java基礎(八)—–深入解析java四種訪問許可權

chen_hao發表於2019-02-21

  Java中的訪問許可權理解起來不難,但完全掌握卻不容易,特別是4種訪問許可權並不是任何時候都可以使用。下面整理一下,在什麼情況下,有哪些訪問許可權可以允許選擇。

一、訪問許可權簡介

訪問許可權控制: 指的是本類及本類內部的成員(成員變數、成員方法、內部類)對其他類的可見性,即這些內容是否允許其他類訪問。

Java 中一共有四種訪問許可權控制,其許可權控制的大小情況是這樣的:public > protected > default(包訪問許可權) > private ,具體的許可權控制看下面表格,列所指定的類是否有許可權允許訪問行的許可權控制下的內容:

訪問許可權 本類 本包的類 子類 非子類的外包類
public
protected
default
private

1、public: 所修飾的類、變數、方法,在內外包均具有訪問許可權;
2、protected: 這種許可權是為繼承而設計的,protected所修飾的成員,對所有子類是可訪問的,但只對同包的類是可訪問的,對外包的非子類是不可以訪問;
3、包訪問許可權(default): 只對同包的類具有訪問的許可權,外包的所有類都不能訪問;
4、private: 私有的許可權,只對本類的方法可以使用;

注意: 要區分開 protected 許可權、包訪問許可權,正確使用它們;

  • 當某個成員能被所有的子類繼承,但不能被外包的非子類訪問,就是用protected;

  • 當某個成員的訪問許可權只對同包的類開放,包括不能讓外包的類繼承這個成員,就用包訪問許可權;

使用訪問許可權控制的原因:

1)使使用者不要碰觸那些他們不該碰觸的部分;
2)類庫設計者可以更改類的內部工作的方式,而不會擔心這樣會對使用者產生重大影響;

二、訪問許可權控制的使用場景

訪問許可權使用的場景可以總結為下面的五種場景,分別對訪問許可權的使用有不同的限制:

1. 外部類的訪問控制

外部類(外部介面) 是相對於內部類(也稱為巢狀類)、內部介面而言的。外部類的訪問控制只能是這兩種:public 、default 。

//public 訪問呢許可權的外部類,所有類都可以使用這個類
public class OuterClass {
}
//default 許可權的外部介面,所有類、介面均可以使用此介面
interface OuterInterface{
}

2. 類裡面的成員的訪問控制

類裡面的成員分為三類 : 成員變數、成員方法、成員內部類(內部介面)

類裡面的成員的訪問控制可以是四種,也就是可以使用所有的訪問控制許可權

public class OuterClass {
    public int aa; //可以被所有的類訪問
    protected boolean bb; //可以被所有子類以及本包的類使用
    void cc() { //default 訪問許可權,能在本包範圍內使用
        System.out.println("包訪問許可權");
    }
    //private許可權的內部類,即這是私有的內部類,只能在本類使用
    private class InnerClass{
    }
}

注意:
這裡的類裡面的成員 是指類的全域性成員,並沒有包括區域性的成員(區域性變數、區域性內部類,沒有區域性內部介面)。或者說,區域性成員是沒有訪問許可權控制的,因為區域性成員只在其所在的作用域內起作用,不可能被其他類訪問到。

public void count(){
    //區域性成員變數
    public int amount;//編譯無法通過,不能用public修飾
    int money;//編譯通過
    //區域性巢狀介面
    class customer{//編譯通過
    }
}

3. 抽象方法的訪問許可權

普通方法是可以使用四種訪問許可權的,但抽象方法是有一個限制:不能用private 來修飾,也即抽象方法不能是私有的,否則,子類就無法繼承實現抽象方法

4. 介面成員的訪問許可權

介面由於其的特殊性,所有成員的訪問許可權都規定得死死的,下面是介面成員的訪問許可權:

  • 變數: public static final

  • 抽象方法: public abstract

  • 靜態方法: public static,JDK1.8後才支援

  • 內部類、內部介面 : public static

也因為所有的一切都預設強制規定好了,所以我們在用的時候,並不一定需要完整寫出所有的修飾符,編譯器會幫我們完成的,也就是,可以少寫修飾符,但不能寫錯修飾符。

public interface Interface_Test {
    public int aa = 6; //少寫了  static final 
    int bb = 5; //
    //巢狀介面,可以不寫public static
     interface cc{
    }
}

5. 構造器的訪問許可權

構造器的訪問許可權可以是以上四種許可權中的任意一種:

1、採用 private:一般是不允許直接構造這個類的物件,再結合工廠方法(static方法),實現單例模式。注意:所有子類都不能繼承它。
2、採用包訪問控制:比較少用,這個類的物件只能在本包中使用,但是如果這個類有static 成員,那麼這個類還是可以在外包使用;(也許可以用於該類的外包單例模式)。
注意:外包的類不能繼承這個類;
3、採用 protected :就是為了能讓所有子類繼承這個類,但是外包的非子類不能訪問這個類;
4、採用 public :對於內外包的所有類都是可訪問的;

注意: 構造方法有點特殊,因為子類的構造器初始化時,都要呼叫父類的構造器,所以一旦父類構造器不能被訪問,那麼子類的構造器呼叫失敗,意味子類繼承父類失敗!

 

相關文章