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 :對於內外包的所有類都是可訪問的;
注意: 構造方法有點特殊,因為子類的構造器初始化時,都要呼叫父類的構造器,所以一旦父類構造器不能被訪問,那麼子類的構造器呼叫失敗,意味子類繼承父類失敗!