Java語言提供了不少修飾符,主要分爲如下兩類:前端
修飾符用來定義類、方法或者變量,一般放在語句的最前端。咱們經過下面的例子來講明:java
public class className { // ... } private boolean myFlag; static final double weeks = 9.5; protected static final int BOXWIDTH = 42; public static void main(String[] arguments) { // 方法體 }
Java中,可使用訪問控制符來保護對類、變量、方法和構造方法的訪問。Java支持4種不一樣的訪問權限。編程
默認的,也稱爲default,在同一包內可見,不使用任何修飾符。函數
私有的,以private修飾符指定,在同一類內可見。this
共有的,以public修飾符指定,對全部類可見。spa
受保護的,以protected修飾符指定,對同一包內的類和全部子類可見。線程
使用默認訪問修飾符聲明的變量和方法,對同一個包內的類是可見的。接口裏的變量都隱式聲明爲public static final,而接口裏的方法默認狀況下訪問權限爲public。orm
實例:對象
以下例所示,變量和方法的聲明能夠不使用任何修飾符。繼承
String version = "1.5.1"; boolean processOrder() { return true; }
私有訪問修飾符是最嚴格的訪問級別,因此被聲明爲private的方法、變量和構造方法只能被所屬類訪問,而且類和接口不能聲明爲private。
聲明爲私有訪問類型的變量只能經過類中公共的getter方法被外部類訪問。
Private訪問修飾符的使用主要用來隱藏類的實現細節和保護類的數據。
下面的類使用了私有訪問修飾符:
public class Logger { private String format; public String getFormat() { return this.format; } public void setFormat(String format) { this.format = format; } }
實例中,Logger類中的format變量爲私有變量,因此其餘類不能直接獲得和設置該變量的值。爲了使其餘類可以操做該變量,定義了兩個public 方法:getFormat() (返回format的值)和setFormat(String)(設置format的值)
被聲明爲public的類、方法、構造方法和接口可以被任何其餘類訪問。
若是幾個相互訪問的public類分佈在不一樣的包中,則須要導入相應public類所在的包。因爲類的繼承性,類全部的公有方法和變量都能被其子類繼承。
如下函數使用了公有訪問控制:
public static void main(String[] arguments) { // ... }
Java程序的main() 方法必須設置成公有的,不然,Java解釋器將不能運行該類。
被聲明爲protected的變量、方法和構造器能被同一個包中的任何其餘類訪問,也可以被不一樣包中的子類訪問。
Protected訪問修飾符不能修飾類和接口,方法和成員變量可以聲明爲protected,可是接口的成員變量和成員方法不能聲明爲protected。
子類能訪問Protected修飾符聲明的方法和變量,這樣就能保護不相關的類使用這些方法和變量。
下面的父類使用了protected訪問修飾符,子類重載了父類的openSpeaker()方法。
class AudioPlayer { protected boolean openSpeaker(Speaker sp) { // 實現細節 } } class StreamingAudioPlayer { boolean openSpeaker(Speaker sp) { // 實現細節 } }
若是把openSpeaker()方法聲明爲private,那麼除了AudioPlayer以外的類將不能訪問該方法。若是把 openSpeaker()聲明爲public,那麼全部的類都可以訪問該方法。若是咱們只想讓該方法對其所在類的子類可見,則將該方法聲明爲 protected。
請注意如下方法繼承的規則:
父類中聲明爲public的方法在子類中也必須爲public。
父類中聲明爲protected的方法在子類中要麼聲明爲protected,要麼聲明爲public。不能聲明爲private。
父類中聲明爲private的方法,不可以被繼承。
爲了實現一些其餘的功能,Java也提供了許多非訪問修飾符。
static修飾符,用來建立類方法和類變量。
Final修飾符,用來修飾類、方法和變量,final修飾的類不可以被繼承,修飾的方法不能被繼承類從新定義,修飾的變量爲常量,是不可修改的。
Abstract修飾符,用來建立抽象類和抽象方法。
Synchronized和volatile修飾符,主要用於線程的編程。
靜態變量:
Static關鍵字用來聲明獨立於對象的靜態變量,不管一個類實例化多少對象,它的靜態變量只有一份拷貝。 靜態變量也被成爲類變量。局部變量能被聲明爲static變量。
靜態方法:
Static關鍵字用來聲明獨立於對象的靜態方法。靜態方法不能使用類的非靜態變量。靜態方法從參數列表獲得數據,而後計算這些數據。
對類變量和方法的訪問能夠直接使用classname.variablename和classname.methodname的方式訪問。
以下例所示,static修飾符用來建立類方法和類變量。
public class InstanceCounter { private static int numInstances = 0; protected static int getCount() { return numInstances; } private static void addInstance() { numInstances++; } InstanceCounter() { InstanceCounter.addInstance(); } public static void main(String[] arguments) { System.out.println("Starting with " + InstanceCounter.getCount() + " instances"); for (int i = 0; i < 500; ++i){ new InstanceCounter(); } System.out.println("Created " + InstanceCounter.getCount() + " instances"); } }
以上實例運行編輯結果以下:
Started with 0 instances Created 500 instances
Final變量:
Final變量能被顯式地初始化而且只能初始化一次。被聲明爲final的對象的引用不能指向不一樣的對象。可是final對象裏的數據能夠被改變。也就是說final對象的引用不能改變,可是裏面的值能夠改變。
Final修飾符一般和static修飾符一塊兒使用來建立類常量。
實例:
public class Test{ final int value = 10; // 下面是聲明常量的實例 public static final int BOXWIDTH = 6; static final String TITLE = "Manager"; public void changeValue(){ value = 12; //將輸出一個錯誤 } }
類中的Final方法能夠被子類繼承,可是不能被子類修改。
聲明final方法的主要目的是防止該方法的內容被修改。
以下所示,使用final修飾符聲明方法。
public class Test{ public final void changeName(){ // 方法體 } }
Final類不能被繼承,沒有類可以繼承final類的任何特性。
實例:
public final class Test { // 類體 }
抽象類:
抽象類不能用來實例化對象,聲明抽象類的惟一目的是爲了未來對該類進行擴充。
一個類不能同時被abstract和final修飾。若是一個類包含抽象方法,那麼該類必定要聲明爲抽象類,不然將出現編譯錯誤。
抽象類能夠包含抽象方法和非抽象方法。
實例:
abstract class Caravan{ private double price; private String model; private String year; public abstract void goFast(); //抽象方法 public abstract void changeColor(); }
抽象方法是一種沒有任何實現的方法,該方法的的具體實現由子類提供。抽象方法不能被聲明成final和strict。
任何繼承抽象類的子類必須實現父類的全部抽象方法,除非該子類也是抽象類。
若是一個類包含若干個抽象方法,那麼該類必須聲明爲抽象類。抽象類能夠不包含抽象方法。
抽象方法的聲明以分號結尾,例如:public abstract sample();
實例:
public abstract class SuperClass{ abstract void m(); //抽象方法 } class SubClass extends SuperClass{ //實現抽象方法 void m(){ ......... } }
Synchronized關鍵字聲明的方法同一時間只能被一個線程訪問。Synchronized修飾符能夠應用於四個訪問修飾符。
實例:
public synchronized void showDetails(){ ....... }
序列化的對象包含被transient修飾的實例變量時,java虛擬機(JVM)跳過該特定的變量。
該修飾符包含在定義變量的語句中,用來預處理類和變量的數據類型。
實例:
public transient int limit = 55; // will not persist public int b; // will persist
Volatile修飾的成員變量在每次被線程訪問時,都強迫從共享內存中重讀該成員變量的值。並且,當成員變量發生變化時,強迫線程將變化值回寫到共享內存。這樣在任什麼時候刻,兩個不一樣的線程老是看到某個成員變量的同一個值。一個volatile對象引用多是null。
實例:
public class MyRunnable implements Runnable { private volatile boolean active; public void run() { active = true; while (active) // line 1 { // 代碼 } } public void stop() { active = false; // line 2 } }
通常地,在一個線程中調用run()方法,在另外一個線程中調用stop()方法。若是line 1中的active位於緩衝區的值被使用,那麼當把line 2中的active設置成false時,循環也不會中止。