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) { }
其中 private, public, protected, static, final 都是修飾符。
1. 訪問控制修飾符
java中,可以使用訪問控制符來保護對類、變量、方法和構造方法的訪問。java 支持 4 種不同的訪問權限。
- default (即默認,什么也不寫): 在同一包內可見,不使用任何修飾符。使用對象:類、接口、變量、方法。
- private : 在同一類內可見。使用對象:變量、方法。 注意:不能修飾類(外部類)
- public : 對所有類可見。使用對象:類、接口、變量、方法
- protected : 對同一包內的類和所有子類可見。使用對象:變量、方法。 注意:不能修飾類(外部類)。
我們可以通過以下表來說明訪問權限:
修飾符 | 當前類 | 同一包內 | 子孫類(同一包) | 子孫類(不同包) | 其他包 |
---|---|---|---|---|---|
public | y | y | y | y | y |
protected | y | y | y | y/n(說明) | n |
default | y | y | y | n | n |
private | y | n | n | n | n |
1)無訪問修飾符
使用默認訪問修飾符聲明的變量和方法,對同一個包內的類是可見的。
接口里的變量都隱式聲明為 public static final,而接口里的方法默認情況下訪問權限為 public。
如下例所示,變量和方法的聲明不使用任何修飾符。
string version = "1.5.1"; boolean processorder() { return true; }
2)私有訪問修飾符 private
私有訪問修飾符是最嚴格的訪問級別,所以被聲明為 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 的值)
3)公有訪問修飾符 public
被聲明為 public 的類、方法、構造方法和接口能夠被任何其他類訪問。
如果幾個相互訪問的 public 類分布在不同的包中,則需要導入相應 public 類所在的包。由于類的繼承性,類所有的公有方法和變量都能被其子類繼承。
以下函數使用了公有訪問控制:
public static void main(string[] arguments) { }
java 程序的 main() 方法必須設置成公有的,否則,java 解釋器將不能運行該類。
4)受保護的訪問修飾符 protected
protected 需要從以下兩個點來分析說明:
- 子類與基類在同一包中:被聲明為 protected 的變量、方法和構造器能被同一個包中的任何其他類訪問;
- 子類與基類不在同一包中:那么在子類中,子類范例可以訪問其從基類繼承而來的 protected 方法,而不能訪問基類范例的protected方法。
protected 可以修飾數據成員,構造方法,方法成員,不能修飾類(內部類除外)。
接口及接口的成員變量和成員方法不能聲明為 protected。
子類能訪問 protected 修飾符聲明的方法和變量,這樣就能保護不相關的類使用這些方法和變量。
下面的父類使用了 protected 訪問修飾符,子類重寫了父類的 openspeaker() 方法。
class audioplayer { protected boolean openspeaker(speaker sp) { // 實現細節 } } class streamingaudioplayer extends audioplayer { protected boolean openspeaker(speaker sp) { // 實現細節 } }
如果把 openspeaker() 方法聲明為 private,那么除了 audioplayer 之外的類將不能訪問該方法。
如果把 openspeaker() 聲明為 public,那么所有的類都能夠訪問該方法。
如果我們只想讓該方法對其所在類的子類可見,則將該方法聲明為 protected。
5. 訪問控制和繼承
請注意以下方法繼承的規則:
- 父類中聲明為 public 的方法在子類中也必須為 public。
- 父類中聲明為 protected 的方法在子類中要么聲明為 protected,要么聲明為 public,不能聲明為 private。
- 父類中聲明為 private 的方法,不能夠被繼承。
6. 非訪問修飾符
為了實現一些其他的功能,java 也提供了許多非訪問修飾符。
static 修飾符,用來修飾類方法和類變量。
final 修飾符,用來修飾類、方法和變量,final 修飾的類不能夠被繼承,修飾的方法不能被繼承類重新定義,修飾的變量為常量,是不可修改的。
abstract 修飾符,用來創建抽象類和抽象方法。
synchronized 和 volatile 修飾符,主要用于線程的編程。
1)static 修飾符
static 變量:
static 關鍵字用來聲明獨立于對象的靜態變量,無論一個類范例化多少對象,它的靜態變量只有一份拷貝。 靜態變量也被稱為類變量。局部變量不能被聲明為 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"); } }
以上范例運行編輯結果如下:
starting with 0 instances created 500 instances
2)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 方法的主要目的是防止該方法的內容被修改。
如下所示,使用 final 修飾符聲明方法。
public class test{ public final void changename(){ // 方法體 } }
final 類:
final 類不能被繼承,沒有類能夠繼承 final 類的任何特性。
public final class test { // 類體 }
3)abstract 修飾符
abstract 類:
抽象類不能用來范例化對象,聲明抽象類的唯一目的是為了將來對該類進行擴充。
一個類不能同時被 abstract 和 final 修飾。如果一個類包含抽象方法,那么該類一定要聲明為抽象類,否則將出現編譯錯誤。
抽象類可以包含抽象方法和非抽象方法。
abstract class caravan{ private double price; private string model; private string year; public abstract void gofast(); //抽象方法 public abstract void changecolor(); }
abstract 方法:
抽象方法是一種沒有任何實現的方法,該方法的的具體實現由子類提供。
抽象方法不能被聲明成 final 和 static。
任何繼承抽象類的子類必須實現父類的所有抽象方法,除非該子類也是抽象類。
如果一個類包含若干個抽象方法,那么該類必須聲明為抽象類。抽象類可以不包含抽象方法。
抽象方法的聲明以分號結尾,例如:public abstract sample();。
public abstract class superclass{ abstract void m(); //抽象方法 } class subclass extends superclass{ //實現抽象方法 void m(){ } }
4)synchronized 修飾符
synchronized 關鍵字聲明的方法同一時間只能被一個線程訪問。synchronized 修飾符可以應用于四個訪問修飾符。
public synchronized void showdetails(){ }
5)transient 修飾符
序列化的對象包含被 transient 修飾的范例變量時,java 虛擬機(jvm)跳過該特定的變量。
該修飾符包含在定義變量的語句中,用來預處理類和變量的數據類型。
public transient int limit = 55; // 不會持久化 public int b; // 持久化
6)volatile 修飾符
volatile 修飾的成員變量在每次被線程訪問時,都強制從共享內存中重新讀取該成員變量的值。而且,當成員變量發生變化時,會強制線程將變化值回寫到共享內存。這樣在任何時刻,兩個不同的線程總是看到某個成員變量的同一個值。
一個 volatile 對象引用可能是 null。
public class myrunnable implements runnable { private volatile boolean active; public void run() { active = true; while (active) // 第一行 { // 代碼 } } public void stop() { active = false; // 第二行 } }
通常情況下,在一個線程調用 run() 方法(在 runnable 開啟的線程),在另一個線程調用 stop() 方法。 如果 第一行 中緩沖區的 active 值被使用,那么在 第二行 的 active 值為 false 時循環不會停止。
但是以上代碼中我們使用了 volatile 修飾 active,所以該循環會停止。