1. 程式人生 > >Java中static關鍵字解析

Java中static關鍵字解析

地方 通過 特性 inf 優化 href compare 筆試 star

Java中的static關鍵字解析

  static關鍵字是很多朋友在編寫代碼和閱讀代碼時碰到的比較難以理解的一個關鍵字,也是各大公司的面試官喜歡在面試時問到的知識點之一。下面就先講述一下static關鍵字的用法和平常容易誤解的地方,最後列舉了一些面試筆試中常見的關於static的考題。以下是本文的目錄大綱:

  一.static關鍵字的用途

  二.static關鍵字的誤區

  三.常見的筆試面試題

  若有不正之處,希望諒解並歡迎批評指正。

  本文轉載自:http://www.cnblogs.com/dolphin0520/p/3799052.html


一. 關鍵字的用途

  在《Java編程思想》P86頁有這樣一段話:

  “static方法就是沒有this的方法。在static方法內部不能調用非靜態方法,反過來是可以的。而且可以在沒有創建任何對象的前提下,僅僅通過類本身來調用static方法。這實際上正是static方法的主要用途。”

  這段話雖然只是說明了static方法的特殊之處,但是可以看出static關鍵字的基本作用,簡而言之,一句話來描述就是:

  方便在沒有創建對象的情況下來進行調用(方法/變量)。

  很顯然,被static關鍵字修飾的方法或者變量不需要依賴於對象來進行訪問,只要類被加載了,就可以通過類名去進行訪問。

  static可以用來修飾類的成員方法、類的成員變量,另外可以編寫static代碼塊來優化程序性能。

1. static方法

  static方法一般稱作靜態方法,由於靜態方法不依賴於任何對象就可以進行訪問,因此對於靜態方法來說,是沒有this的,因為它不依附於任何對象,既然都沒有對象,就談不上this了。並且由於這個特性,在靜態方法中不能訪問類的非靜態成員變量和非靜態成員方法,因為非靜態成員方法/變量都是必須依賴具體的對象才能夠被調用。

  但是要註意的是,雖然在靜態方法中不能訪問非靜態成員方法和非靜態成員變量,但是在非靜態成員方法中是可以訪問靜態成員方法/變量的。舉個簡單的例子:

  技術分享圖片

  在上面的代碼中,由於print2方法是獨立於對象存在的,可以直接用過類名調用。假如說可以在靜態方法中訪問非靜態方法/變量的話,那麽如果在main方法中有下面一條語句:

  MyObject.print2();

  此時對象都沒有,str2根本就不存在,所以就會產生矛盾了。同樣對於方法也是一樣,由於你無法預知在print1方法中是否訪問了非靜態成員變量,所以也禁止在靜態成員方法中訪問非靜態成員方法。

  而對於非靜態成員方法,它訪問靜態成員方法/變量顯然是毫無限制的。

  因此,如果說想在不創建對象的情況下調用某個方法,就可以將這個方法設置為static。我們最常見的static方法就是main方法,至於為什麽main方法必須是static的,現在就很清楚了。因為程序在執行main方法的時候沒有創建任何對象,因此只有通過類名來訪問

2. static變量

  static變量也稱作靜態變量,靜態變量和非靜態變量的區別是:靜態變量被所有的對象所共享,在內存中只有一個副本,它當且僅當在類初次加載時會被初始化。而非靜態變量是對象所擁有的,在創建對象的時候被初始化,存在多個副本,各個對象擁有的副本互不影響

  static成員變量的初始化順序按照定義的順序進行初始化

3.static代碼塊

  static關鍵字還有一個比較關鍵的作用就是 用來形成靜態代碼塊以優化程序性能。static塊可以置於類中的任何地方,類中可以有多個static塊。在類初次被加載的時候,會按照static塊的順序來執行每個static塊,並且只會執行一次。

  為什麽說static塊可以用來優化程序性能,是因為它的特性:只會在類加載的時候執行一次。下面看個例子:

 1 class Person{
 2     private Date birthday;
 3 
 4     Person(Date birthday){
 5         this.birthday = birthday;
 6     }
 7 
 8     boolean isBornBoomer(){
 9         Date startDate = new Date(1956,1,1);
10         Date endDate = new Date(1964,12,12);
11         int a = birthday.compareTo(startDate);  //返回值0  1   -1  ==  >   <
12         int b = birthday.compareTo(endDate);
13         return a>=0 && b<=0;
14     }
15 }

  isBornBoomer是用來這個人是否是1946-1964年出生的,而每次isBornBoomer被調用的時候,都會生成startDate和birthDate兩個對象,造成了空間浪費,如果改成這樣效率會更好:

 1 class Person{
 2     private Date birthday;
 3     private static Date startDate,endDate;
 4     static {
 5         System.out.println("Initialing Class Object Person");
 6         startDate = new Date(1956,1,1);
 7         endDate = new Date(1964,12,12);
 8     }
 9 
10     public Person(Date birthday) {
11         this.birthday = birthday;
12     }
13 
14     boolean isBornBoomer() {
15         return birthday.compareTo(startDate)>=0 && birthday.compareTo(endDate) < 0;
16     }
17 }

二. static關鍵字的誤區

  1.static關鍵字會改變類中成員的訪問權限嗎?

  有些初學的朋友會將java中的static與C/C++中的static關鍵字的功能混淆了。在這裏只需要記住一點:與C/C++中的static不同,Java中的static關鍵字不會影響到變量或者方法的作用域。在Java中能夠影響到訪問權限的只有private、public、protected(包括包訪問權限)這幾個關鍵字。看下面的例子就明白了:

  技術分享圖片

  2.能通過this訪問靜態成員變量嗎?

  雖然對於靜態方法來說沒有this,那麽在非靜態方法中能夠通過this訪問靜態成員變量嗎?先看下面的一個例子,這段代碼輸出的結果是什麽?

 1 public class Main {  
 2     static int value = 33;
 3  
 4     public static void main(String[] args) throws Exception{
 5         new Main().printValue();
 6     }
 7  
 8     private void printValue(){
 9         int value = 3;
10         System.out.println(this.value);
11     }
12 }

  

技術分享圖片
33
View Code

  這裏面主要考察隊this和static的理解。this代表什麽?this代表當前對象,那麽通過new Main()來調用printValue的話,當前對象就是通過new Main()生成的對象。而static變量是被對象所享有的,因此在printValue中的this.value的值毫無疑問是33。在printValue方法內部的value是局部變量,根本不可能與this關聯,所以輸出結果是33。在這裏永遠要記住一點:靜態成員變量雖然獨立於對象,但是不代表不可以通過對象去訪問,所有的靜態方法和靜態變量都可以通過對象訪問(只要訪問權限足夠)。

  3.static能作用於局部變量麽?

  在C/C++中static是可以作用域局部變量的,但是在Java中切記:static是不允許用來修飾局部變量。不要問為什麽,這是Java語法的規定。

三. 常見的筆試面試題

  下面列舉一些面試筆試中經常遇到的關於static關鍵字的題目,僅供參考,如有補充歡迎下方留言。

  1.下面這段代碼的輸出結果是什麽?

 1 public class Test extends Base{
 2  
 3     static{
 4         System.out.println("test static");
 5     }
 6      
 7     public Test(){
 8         System.out.println("test constructor");
 9     }
10      
11     public static void main(String[] args) {
12         new Test();
13     }
14 }
15  
16 class Base{
17      
18     static{
19         System.out.println("base static");
20     }
21      
22     public Base(){
23         System.out.println("base constructor");
24     }
25 }

技術分享圖片
1 base static
2 test static
3 base constructor
4 test constructor
View Code

  至於為什麽是這個結果,我們先不討論,先來想一下這段代碼具體的執行過程,在執行開始,先要尋找到main方法,因為main方法是程序的入口,但是在執行main方法之前,必須先加載Test類,而在加載Test類的時候發現Test類繼承自Base類,因此會轉去先加載Base類,在加載Base類的時候,發現有static塊,便執行了static塊。在Base類加載完成之後,便繼續加載Test類,然後發現Test類中也有static塊,便執行static塊。

  在加載完所需的類之後,便開始執行main方法。在main方法中執行new Test()的時候會先調用父類的構造器,然後再調用自身的構造器。因此,便出現了上面的輸出結果。

2.這段代碼的輸出結果是什麽?

Java中static關鍵字解析