第六章 訪問權(quán)限控制
1.訪問權(quán)限控制的等級(jí)
- 從最大權(quán)限到最小權(quán)限依次為:public八堡、protected型豁、包訪問權(quán)限(沒有關(guān)鍵字)和private捏检。
2.包:庫(kù)單元
Java用package關(guān)鍵字將構(gòu)件捆綁到一個(gè)內(nèi)聚的類庫(kù)單元中取募。當(dāng)編寫一個(gè)Java源代碼文件時(shí)宰译,此文件通常被稱為編譯單元(有時(shí)也被稱為轉(zhuǎn)譯單元)檐蚜。每個(gè)編譯單元必須有一個(gè)后綴名.java,而在編譯單元內(nèi)則可以有一個(gè)public類沿侈,該類的名稱必須與文件的名稱相同(包括大小寫闯第,但不包括文件的后綴名.java)。每個(gè)編譯單元只能有一個(gè)public類缀拭,否則編譯器就不會(huì)接受咳短。如果在該編譯單元之中還有額外的類的話,那么在包之外的世界是無(wú)法看到這些類的蛛淋,這是因?yàn)樗鼈儾皇莗ublic類咙好,而且它們主要用來(lái)為主public類提供支持。
- package語(yǔ)句必須是文件中除注釋以外的第一句程序代碼褐荷。
- Java包的命名規(guī)則全部使用小寫字母勾效,包括中間的字也是如此。
Java解釋器的運(yùn)行過(guò)程如下:
The Java interpreter proceeds as follows. First, it finds the environment variable CLASSPATH (set via the operating system, and sometimes by the installation program that installs Java or a Java-based tool on your machine). CLASSPATH contains one or more directories that are used as roots in a search for .class files. Starting at that root, the interpreter will take the package name and replace each dot with a slash to generate a path name off of the CLASSPATH root (so package foo.bar.baz becomes foo\bar\baz or foo/bar/baz or possibly something else, depending on your operating system). This is then concatenated to the various entries in the CLASSPATH. That’s where it looks for the .class file with the name corresponding to the class you’re trying to create. (It also searches some standard directories relative to where the Java interpreter resides.)
3.Java訪問權(quán)限修飾詞
訪問權(quán)限修飾詞可以類中每個(gè)成員的定義之前(無(wú)論它是一個(gè)域還是一個(gè)方法)。
包訪問權(quán)限
- 默認(rèn)訪問權(quán)限沒有任何關(guān)鍵字层宫,但是通常是指包訪問權(quán)限(有時(shí)也表示成為friendly)绘迁。其意味著當(dāng)前的包中所有其他類對(duì)那個(gè)成員都有訪問權(quán)限,但是對(duì)于這個(gè)包之外的所有類卒密,這個(gè)成員卻是private缀台。由于一個(gè)編譯單元(即一個(gè)文件),只能隸屬于一個(gè)包哮奇,所有經(jīng)由包訪問權(quán)限膛腐,處于同一個(gè)編譯單元中的所有類彼此之間都是自動(dòng)可訪問的。
取得對(duì)某個(gè)成員的訪問權(quán)的唯一途徑是:
注意:同處于相同的目錄但是沒有給自己設(shè)定任何包名稱的Java文件將被看做是隸屬于該目錄的默認(rèn)包之中鼎俘。
All the files within the same directory that don’t have explicit package declarations are implicitly part of the default package for
that directory.
public:接口訪問權(quán)限
使用關(guān)鍵字public哲身,就意味著public之后緊跟著的成員聲明自己對(duì)每個(gè)人(任何類)都是可用的(可訪問的),尤其是使用類庫(kù)的客戶程序員更是如此贸伐。
private:你無(wú)法訪問
關(guān)鍵字private的意思是勘天,除了包含該成員的類之外,其他任何類都無(wú)法訪問這個(gè)成員捉邢。
注意:使用類的客戶端程序員是無(wú)法訪問包訪問權(quán)限成員的脯丝。
private的運(yùn)用示例:可能想控制如何創(chuàng)建對(duì)象,并阻止別人直接訪問某個(gè)特定的構(gòu)造器(或全部構(gòu)造器)伏伐。示例如下:
class Sundae {
private Sundae() {}
static Sundae makeASundae() {
return new Sundae();
}
}
public class IceCream {
public static void main(String[] args) {
//! Sundae x = new Sundae();
Sundae x = Sundae.makeASundae();
}
}
上述示例中宠进,不能通過(guò)構(gòu)造器來(lái)創(chuàng)建Sundae對(duì)象,而必須調(diào)用makeASundae()方法來(lái)達(dá)到此目的藐翎。
- 說(shuō)明:除非必須公開底層實(shí)現(xiàn)細(xì)目(此種境況很少)材蹬,否則就應(yīng)該將所有的域指定為private。
- However, just because a reference to an object is private inside a class doesn’t mean that some other object can’t have a public reference to the same object.(如何理解此句話吝镣?)
protected:繼承訪問權(quán)限
- 如果創(chuàng)建了一個(gè)新包堤器,并在該新包中創(chuàng)建一個(gè)類(子類)繼承另一個(gè)包中的某個(gè)類(基類)裳瘪,那么唯一可以訪問基類的成員就是其public成員和protected成員(當(dāng)然挺举,如果在同一個(gè)包內(nèi)執(zhí)行繼承工作贪壳,就可以操縱所有的擁有包訪問權(quán)限的成員)决侈。
- protected也提供包訪問權(quán)限,即相同包內(nèi)的其他所有類可以訪問protected元素术瓮。
小結(jié):訪問權(quán)限示意圖
4.接口和實(shí)現(xiàn)
- 訪問權(quán)限的控制常被稱為是具體實(shí)現(xiàn)的隱藏。把數(shù)據(jù)和方法包裝進(jìn)類中,以及具體實(shí)現(xiàn)的隱藏裕膀,行共同被稱作是封裝。其結(jié)果是一個(gè)同時(shí)帶有特征和行為的數(shù)據(jù)類型勇哗。
- 為了清楚起見昼扛,可能會(huì)采用一種將public成員置于開頭,后面緊跟protected、包訪問權(quán)限和private成員的創(chuàng)建類形式抄谐。示例如下:
public class OrganizedByAccess {
public void pub1() { /* ... */ }
public void pub2() { /* ... */ }
public void pub3() { /* ... */ }
private void priv1() { /* ... */ }
private void priv2() { /* ... */ }
private void priv3() { /* ... */ }
private int i;
// ...
}
5.類的訪問權(quán)限
- 為了控制某個(gè)類的訪問權(quán)限渺鹦,修飾詞必須出現(xiàn)于關(guān)鍵字class之前,如蛹含,
public class Widget{...}
毅厚。
類的一些限制如下:
- 類既不可以是private(否則,除該類外浦箱,其他任何類都不可以訪問它)吸耿,也不可以是protected的。所以對(duì)類的訪問權(quán)限酷窥,僅有兩個(gè)選擇:包訪問權(quán)限或public咽安。如果不希望其他人對(duì)該類用于訪問權(quán)限,可以把所以的構(gòu)造器都指定為private蓬推,從而阻止任何人創(chuàng)建該類的對(duì)象妆棒,但是有一個(gè)例外,就是你在該類的static成員內(nèi)部可以創(chuàng)建該類的對(duì)象沸伏。示例如下:
class Soup1 {
private Soup1() {}
// (1) Allow creation via static method:
public static Soup1 makeSoup() {
return new Soup1();
}
}
class Soup2 {
private Soup2() {}
// (2) Create a static object and return a reference
// upon request.(The "Singleton" pattern):
private static Soup2 ps1 = new Soup2();
public static Soup2 access() {
return ps1;
}
public void f() {}
}
// Only one public class allowed per file:
public class Lunch {
void testPrivate() {
// Can't do this! Private constructor:
//! Soup1 soup = new Soup1();
}
void testStatic() {
Soup1 soup = Soup1.makeSoup();
}
void testSingleton() {
Soup2.access().f();
}
}
說(shuō)明:
1)如果想要在返回引用前在Soup1上做一些額外的工作糕珊,或是如果想要記錄到底創(chuàng)建了多少個(gè)Soup1對(duì)象(可能與限制其數(shù)量),這種做法將會(huì)大有裨益的毅糟。
2)Soup2用到了單例模式(singleton)放接,這是因?yàn)槲覀兪冀K只能創(chuàng)建它的一個(gè)對(duì)象。
-
注意:
1)如果沒有為類訪問權(quán)限指定一個(gè)訪問修飾符留特,它就會(huì)默認(rèn)得到包訪問權(quán)限纠脾。此時(shí),即使將該類的構(gòu)造器聲明為public的蜕青,也不能在包外實(shí)例化該類苟蹈。并且,即使該類中擁有static public字段或static public方法右核,也不能在包外直接通過(guò)該類訪問慧脱。總之贺喝,在具有包訪問權(quán)限的類中菱鸥,使用public并不能將其修飾的成員或構(gòu)造器提升到public權(quán)限,即使用public無(wú)效躏鱼。2)一個(gè)類的默認(rèn)構(gòu)造器的訪問權(quán)限和該類的訪問權(quán)限相同氮采。
3)內(nèi)部類可以是private或protected的。