據說一半以上的java程序員會出錯的題

大三即將結束,經過一段時間的較爲系統的自學java, 突然看到一個自稱一半以上的java程序員都會出錯的程序尤爲感興趣。便深究了一番

     程序代碼如下:


  1. package com.longpo;  
  •   
  • class

 Person {  

  •     private

 static Person person = new Person();  

  •   
  •     public

 static int count1;  

  •   
  •     public

 static int count2 = 5;  

  •   
  •     private

 Person() {  

  •         count1++;  
  •         count2++;  
  •     }  
  •       
  •     public

 static Person getInstance()  

  •     {  
  •         return

 person;  

  •     }  
  •       
  • }  
  •   
  • public

 class Testsingleton {  

  •     public

 static void main(String[] args) {  

  •   
  •         Person person=Person.getInstance();  
  •           
  •         //可以用直接Person.count1

  

  •         System.out.println("count1: "

+person.count1);  

  •       
  •         System.out.println("count2: "

+person.count2);  

  •     }  
  •   
  • }  

上面代碼輸出的結果是什麼呢?

很容易讓人覺得會輸出1和6(從題目可知肯定不是這麼簡單),那時我猜答案應該是1和5,但說不出所以然。我把代碼賦值到Eclipse下運行。得到

 

  

    和我猜的一樣,可是我完全不知道其原因。。於是就開始了谷歌求知之路,經過研究,ClassLoader漸漸映入我的眼簾。下面我來說說我認爲的原因,有錯誤還望指導更正

類在執行之前會執行三個步驟:

1.類的加載:查找並加載類的二進制數據,把對應的class文件加載到內存

2.連接

     2.1. 驗證:確保被加載的類的正確性(主要防止噁心的class文件被加載)

     2.2. 準備:爲類的靜態變量分配內存,並將其初始化爲默認值

     2.3. 解析:把類中的符合引用轉換爲直接引用

3.初始化:爲類的靜態變量賦予正確的初始值

 

發現其中步驟2.2和3提到了關鍵字靜態變量,重點關注這兩步,2.2的結果會導致爲靜態變量

person,count1,count2分配內存並賦值(默認值)

person=null;

count1=count2=0

 

到步驟3,初始化時會爲靜態變量賦予正確的值,那麼什麼時候纔會進行初始化呢?接着

谷歌百度,得到:

     所有的java虛擬機實現必須在每一個類或接口被java程序“首次主動使用

”時才初始化

 

java對類的使用方式分爲:主動使用,被動使用

主動使用有六種:(除這6種外,其他都是被動使用)

1。創建類的實例

2。訪問某個類或接口的靜態變量或對該靜態變量賦值

3。調用類的靜態方法

4。反射

5。初始化類的子類

6。java虛擬機啓動時被標註位啓動類的類

 

懂了第三步(初始化),那麼就可以來解釋程序爲什麼輸出1和5了,我畫了一個圖

 

現在應該明白了其中的原因了吧,爲了檢驗是否真的明白,可以把代碼再改爲

 

  1. package

 com.longpo;  

  •   
  • class

 Person {  

  •       
  •   
  •     public

 static int count1;  

  •       
  •   
  •     public

 static int count2 = 5;  

  •       
  •     private

 static Person person = new Person();  

  •       
  •     private

 Person() {  

  •         count1++;  
  •         count2++;  
  •     }  
  •       
  •     public

 static Person getInstance()  

  •     {  
  •         return

 person;  

  •     }  
  •       
  • }  
  •   
  • public

 class Testsingleton {  

  •     public

 static void main(String[] args) {  

  •   
  •         Person person=Person.getInstance();  
  •           
  •         //可以用直接Person.count1

  

  •         System.out.println("count1: "

+person.count1);  

  •       
  •         System.out.println("count2: "

+person.count2);  

  •     }  
  •   
  • }  ​

原文來自techfox技術論壇java社區 http://techfoxbbs.com/blog-1-3.html

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章