天天看點

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 修飾符指定,在同一類内可見。

共有的,以 public 修飾符指定,對所有類可見。

受保護的,以 protected 修飾符指定,對同一包内的類和所有子類可見。

預設通路修飾符-不使用任何關鍵字

使用預設通路修飾符聲明的變量和方法,對同一個包内的類是可見的。接口裡的變量都隐式聲明為public static final,而接口裡的方法預設情況下通路權限為public。

執行個體:

如下例所示,變量和方法的聲明可以不使用任何修飾符。

String version = "1.5.1";
boolean processOrder() {
   return true;
}


      

Java 修飾符

私有通路修飾符-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的值)

公有通路修飾符-public

被聲明為public的類、方法、構造方法和接口能夠被任何其他類通路。

如果幾個互相通路的public類分布在不同的包中,則需要導入相應public類所在的包。由于類的繼承性,類所有的公有方法和變量都能被其子類繼承。

以下函數使用了公有通路控制:

public static void main(String[] arguments) {
   // ...
}      

Java程式的main() 方法必須設定成公有的,否則,Java解釋器将不能運作該類。

受保護的通路修飾符-protected

被聲明為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變量。

  • 靜态方法:

    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對象的引用不能改變,但是裡面的值可以改變。

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 {
   // 類體
}      

Abstract修飾符

抽象類:

抽象類不能用來執行個體化對象,聲明抽象類的唯一目的是為了将來對該類進行擴充。

一個類不能同時被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關鍵字聲明的方法同一時間隻能被一個線程通路。Synchronized修飾符可以應用于四個通路修飾符。

public synchronized void showDetails(){
.......
}      

Transient修飾符

序列化的對象包含被transient修飾的執行個體變量時,java虛拟機(JVM)跳過該特定的變量。

該修飾符包含在定義變量的語句中,用來預處理類和變量的資料類型。

public transient int limit = 55;   // will not persist
public int b; // will persist      

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; // 第二行
    }
}