大三即將結束,經過一段時間的較爲系統的自學java, 突然看到一個自稱一半以上的java程序員都會出錯的程序尤爲感興趣。便深究了一番
程序代碼如下:
- 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了,我畫了一個圖
現在應該明白了其中的原因了吧,爲了檢驗是否真的明白,可以把代碼再改爲
- 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