摘錄自http://www.runoob.com/java/java-modifier-types.html
Java 修飾符
Java語言提供了很多修飾符,主要分為以下兩類:
訪問修飾符
非訪問修飾符
修飾符用來定義類、方法或者變量,通常放在語句的最前端。我們通過下面的例子來說明:
publicclassclassName {// ...}privatebooleanmyFlag;staticfinaldoubleweeks =9.5;protectedstaticfinalintBOXWIDTH =42;publicstaticvoidmain(String[]arguments){// 方法體}
訪問控制修飾符
Java中,可以使用訪問控制符來保護對類、變量、方法和構造方法的訪問。Java支持4種不同的訪問權限。
默認的,也稱為?default,在同一包內可見,不使用任何修飾符。
私有的,以?private?修飾符指定,在同一類內可見。
共有的,以?public?修飾符指定,對所有類可見。
受保護的,以?protected?修飾符指定,對同一包內的類和所有子類可見。
默認訪問修飾符-不使用任何關鍵字
使用默認訪問修飾符聲明的變量和方法,對同一個包內的類是可見的。接口里的變量都隱式聲明為public static final,而接口里的方法默認情況下訪問權限為public。
實例:
如下例所示,變量和方法的聲明可以不使用任何修飾符。
Stringversion ="1.5.1";booleanprocessOrder(){returntrue;}
私有訪問修飾符-private
私有訪問修飾符是最嚴格的訪問級別,所以被聲明為private的方法、變量和構造方法只能被所屬類訪問,并且類和接口不能聲明為private。
聲明為私有訪問類型的變量只能通過類中公共的getter方法被外部類訪問。
Private訪問修飾符的使用主要用來隱藏類的實現細節和保護類的數據。
下面的類使用了私有訪問修飾符:
publicclassLogger{privateStringformat;publicStringgetFormat(){returnthis.format;}publicvoidsetFormat(Stringformat){this.format =format;}}
實例中,Logger類中的format變量為私有變量,所以其他類不能直接得到和設置該變量的值。為了使其他類能夠操作該變量,定義了兩個public方法:getFormat() (返回format的值)和setFormat(String)(設置format的值)
公有訪問修飾符-public
被聲明為public的類、方法、構造方法和接口能夠被任何其他類訪問。
如果幾個相互訪問的public類分布在不同的包中,則需要導入相應public類所在的包。由于類的繼承性,類所有的公有方法和變量都能被其子類繼承。
以下函數使用了公有訪問控制:
publicstaticvoidmain(String[]arguments){// ...}
Java程序的main() 方法必須設置成公有的,否則,Java解釋器將不能運行該類。
受保護的訪問修飾符-protected
被聲明為protected的變量、方法和構造器能被同一個包中的任何其他類訪問,也能夠被不同包中的子類訪問。
Protected訪問修飾符不能修飾類和接口,方法和成員變量能夠聲明為protected,但是接口的成員變量和成員方法不能聲明為protected。
子類能訪問Protected修飾符聲明的方法和變量,這樣就能保護不相關的類使用這些方法和變量。
下面的父類使用了protected訪問修飾符,子類重載了父類的openSpeaker()方法。
classAudioPlayer{protectedbooleanopenSpeaker(Speakersp){// 實現細節}}classStreamingAudioPlayer{booleanopenSpeaker(Speakersp){// 實現細節}}
如果把openSpeaker()方法聲明為private,那么除了AudioPlayer之外的類將不能訪問該方法。如果把openSpeaker()聲明為public,那么所有的類都能夠訪問該方法。如果我們只想讓該方法對其所在類的子類可見,則將該方法聲明為protected。
訪問控制和繼承
請注意以下方法繼承的規則:(父類為整體約束,子類不能比父類更嚴格)
父類中聲明為public的方法在子類中也必須為public。
父類中聲明為protected的方法在子類中要么聲明為protected,要么聲明為public。不能聲明為private。
父類中聲明為private的方法,不能夠被繼承。
非訪問修飾符
為了實現一些其他的功能,Java也提供了許多非訪問修飾符。
static修飾符,用來創建類方法和類變量。
Final修飾符,用來修飾類、方法和變量,final修飾的類不能夠被繼承,修飾的方法不能被繼承類重新定義,修飾的變量為常量,是不可修改的。
Abstract修飾符,用來創建抽象類和抽象方法。
Synchronized和volatile修飾符,主要用于線程的編程。
Static修飾符
靜態變量:
Static關鍵字用來聲明獨立于對象的靜態變量,無論一個類實例化多少對象,它的靜態變量只有一份拷貝。 靜態變量也被成為類變量。局部變量不能被聲明為static變量。
靜態方法:
Static關鍵字用來聲明獨立于對象的靜態方法。靜態方法不能使用類的非靜態變量(因為靜態方法可以不用創建對象就調用,非靜態方法必須有了對象的實例才能調用)。靜態方法從參數列表得到數據,然后計算這些數據。
只要記住:
用statci修飾的成員是屬于類的,在static的方法里可以用類名直接調用;
不用statci修飾的成員是屬于具體實例對象的,需要用對象名調用,且在static的方法里不可以調用。
因為靜態方法的調用不是通過實例對象進行的,所以在靜態方法中沒有this指針,不能訪問所屬類的非靜態變量和方法,只能訪問方法體內的局部變量、自己的參數和靜態變量。所以你在在static的主函數里調用非static的方法時肯定報錯。
對類變量和方法的訪問可以直接使用classname.variablename和classname.methodname的方式訪問。
如下例所示,static修飾符用來創建類方法和類變量。
publicclassInstanceCounter{privatestaticintnumInstances =0;protectedstaticintgetCount(){returnnumInstances;}privatestaticvoidaddInstance(){numInstances++;}InstanceCounter(){InstanceCounter.addInstance();}publicstaticvoidmain(String[]arguments){System.out.println("Starting with "+InstanceCounter.getCount()+" instances");for(inti =0;i <500;++i){newInstanceCounter();}System.out.println("Created "+InstanceCounter.getCount()+" instances");}}
以上實例運行編輯結果如下:
Startedwith0instances
Created500instances
Final修飾符
Final變量:
Final變量能被顯式地初始化并且只能初始化一次。被聲明為final的對象的引用不能指向不同的對象。但是final對象里的數據可以被改變。也就是說final對象的引用不能改變,但是里面的值可以改變。
Final修飾符通常和static修飾符一起使用來創建類常量。
實例:
publicclassTest{finalintvalue =10;// 下面是聲明常量的實例publicstaticfinalintBOXWIDTH =6;staticfinalStringTITLE ="Manager";publicvoidchangeValue(){value =12;//將輸出一個錯誤}}
Final方法
類中的Final方法可以被子類繼承,但是不能被子類修改。
聲明final方法的主要目的是防止該方法的內容被修改。
如下所示,使用final修飾符聲明方法。
publicclassTest{publicfinalvoidchangeName(){// 方法體}}
Final類
Final類不能被繼承,沒有類能夠繼承final類的任何特性。
實例:
publicfinalclassTest{// 類體}
Abstract修飾符
抽象類:
抽象類不能用來實例化對象,聲明抽象類的唯一目的是為了將來對該類進行擴充。
一個類不能同時被abstract和final修飾(abstract是抽象類,指定必須要子類繼承才可以使用,而final是指定不允許繼承)。如果一個類包含抽象方法,那么該類一定要聲明為抽象類,否則將出現編譯錯誤。
抽象類可以包含抽象方法和非抽象方法。
實例:
abstractclassCaravan{privatedoubleprice;privateStringmodel;privateStringyear;publicabstractvoidgoFast();//抽象方法publicabstractvoidchangeColor();}
抽象方法
抽象方法是一種沒有任何實現的方法,該方法的的具體實現由子類提供。抽象方法不能被聲明成final(因為類中的Final方法可以被子類繼承,但是不能被子類修改)和strict。
任何繼承抽象類的子類必須實現父類的所有抽象方法,除非該子類也是抽象類。
如果一個類包含若干個抽象方法,那么該類必須聲明為抽象類。抽象類可以不包含抽象方法。
抽象方法的聲明以分號結尾,例如:public abstract sample();
實例:
publicabstractclassSuperClass{abstractvoidm();//抽象方法}classSubClassextendsSuperClass{//實現抽象方法voidm(){.........}}
Synchronized修飾符
Synchronized關鍵字聲明的方法同一時間只能被一個線程訪問。Synchronized修飾符可以應用于四個訪問修飾符。
實例:
publicsynchronizedvoidshowDetails(){.......}
Transient修飾符
序列化的對象包含被transient修飾的實例變量時,java虛擬機(JVM)跳過該特定的變量。
該修飾符包含在定義變量的語句中,用來預處理類和變量的數據類型。
實例:
publictransientintlimit =55;// will not persistpublicintb;// will persist
Volatile修飾符
Volatile 修飾的成員變量在每次被線程訪問時,都強制從共享內存中重新讀取該成員變量的值。而且,當成員變量發生變化時,會強制線程將變化值回寫到共享內存。這樣在任何時刻,兩個不同的線程總是看到某個成員變量的同一個值。
一個volatile對象引用可能是null。
實例:
publicclassMyRunnableimplementsRunnable{privatevolatilebooleanactive;publicvoidrun(){active =true;while(active)// 第一行{// 代碼}}publicvoidstop(){active =false;// 第二行}}
通常情況下,在一個線程調用 run() 方法(在 Runnable 開啟的線程),在另一個線程調用 stop() 方法。 如果?第一行中緩沖區的 active 值被使用,那么在?第二行?的 active 值為 false 時循環不會停止。
但是以上代碼中我們使用了 volatile 修飾 active,所以該循環會停止。