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