亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁技術文章
文章詳情頁

Java類的實例化順序

瀏覽:91日期:2024-02-10 11:57:39

問題描述

在驗證《Core Java》第9版4-5代碼時,發現程序輸出結果和自己理解的不太一樣。

import java.util.Random;class Employee { private static int nextId; private int id; private String name = ''; private double salary; static {Random generator = new Random();nextId = generator.nextInt(10000); } {id = nextId;nextId++; } public Employee(String name, double salary) {this.name = name;this.salary = salary; } public Employee(double salary) {this('Employee #' + nextId, salary); } public Employee() { } public String getName() {return name; } public double getSalary() {return salary; } public int getId() {return id; }}public class ConstructorTest { public static void main(String[] args) {Employee[] staff = new Employee[3];staff[0] = new Employee('Harry', 40000);staff[1] = new Employee(60000);staff[2] = new Employee();for (Employee e : staff) { System.out.println('id = ' + e.getId() + ', name = ' + e.getName() + ', salary = ' + e.getSalary());} }}

以下是輸出結果:

id = 6943, name = Harry, salary = 40000.0id = 6944, name = Employee #6944, salary = 60000.0id = 6945, name = , salary = 0.0

根據第一條語句得出靜態初始化塊生成的nextId為6943,然后在初始化塊中id被賦值為6943,nextId自增后為6944。再執行第一個構造函數;

那么對于第二個對象來說,就應該直接執行初始化塊,此時id為6944,nextId自增為6945。再執行第二個構造函數,此時this('Employee #' + nextId, salary);語句中的nextId應該為6945,為什么輸出結果為6944呢?

問題解答

回答1:

這個類初始化的順序確實是個神奇的問題,只可根據結果去理解。我打了個斷點去測試,staff[0] = new Employee('Harry', 40000);和staff[2] = new Employee();都是代碼塊先于構造方法執行,但staff[1] = new Employee(60000);卻先執行走到this('Employee #' + nextId, salary);,然后代碼塊,然后public Employee(String name, double salary)構造函數。如果你使用2,則按你的預期,代碼塊先于構造方法。

public Employee(double salary) { // 1 this('Employee #' + nextId, salary); // 2// this.name = 'Employee #' + nextId; // this.salary = salary;}回答2:

正常來說,java 編譯器會把實例初始化塊復制構造方法中,具體位置在調用父類的構造方法以后,構造方法里面的語句之前,但是存在例外情況。Java 官方的 Tutorials 里說初始化塊會被復制到每個構造方法里面其實是不嚴謹的。

具體到這個例子,需要考慮一個問題,如果編譯器把初始化塊復制到每個構造方法里面,那么對于在構造方法里面調用了其他構造方法的情況,這個初始化塊就會執行兩次,就像例子里面的

public Employee(double salary) {this('Employee #' + nextId, salary); // 調用了另一個構造方法}

如果編譯器把初始化塊里的代碼復制到了public Employee(double salary)和public Employee(String name, double salary)里面,這個初始化塊就會執行兩次,為了避免這種情況,編譯器作了一個簡單的處理,編譯器發現public Employee(double salary)調用了本類的另一個構造方法,就沒有把初始化塊的代碼拷貝到這個構造方法里面。也就是說在初始化第二個對象的時候,這個初始化塊是推遲到調用this('Employee #' + nextId, salary);后,在執行Employee(String name, double salary)的時候才執行的,由于推遲了初始化塊的執行,在決定傳遞的參數 nextId 的時候,仍然是未自增的值。如果把這個構造方法修改為

public Employee(double salary) { // this('Employee #' + nextId, salary); this.name = 'Employee #' + nextId; this.salary = salary;}

輸出結果就會變為

id = 5473, name = Harry, salary = 40000.0id = 5474, name = Employee #5475, salary = 60000.0id = 5475, name = , salary = 0.0

而修改之前的情況,反編譯下 class 文件就能看出來編譯器最后的輸出結果,這里只貼三個構造方法,可以很明顯的看出來,第二個構造方法并沒有被復制初始化塊的內容,直接調用了另一個構造方法。

public Employee(java.lang.String, double); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object.'<init>':()V 4: aload_0 5: ldc #2 // String 7: putfield #3 // Field name:Ljava/lang/String; 10: aload_0 11: getstatic #4 // Field nextId:I 14: putfield #5 // Field id:I 17: getstatic #4 // Field nextId:I 20: iconst_1 21: iadd 22: putstatic #4 // Field nextId:I 25: aload_0 26: aload_1 27: putfield #3 // Field name:Ljava/lang/String; 30: aload_0 31: dload_2 32: putfield #6 // Field salary:D 35: return public Employee(double); Code: 0: aload_0 1: new #7 // class java/lang/StringBuilder 4: dup 5: invokespecial #8 // Method java/lang/StringBuilder.'<init>':()V 8: ldc #9 // String Employee # 10: invokevirtual #10 // Method java/lang/StringBuilder.append:(Ljava/lang/String;)Ljava/lang/StringBuilder; 13: getstatic #4 // Field nextId:I 16: invokevirtual #11 // Method java/lang/StringBuilder.append:(I)Ljava/lang/StringBuilder; 19: invokevirtual #12 // Method java/lang/StringBuilder.toString:()Ljava/lang/String; 22: dload_1 23: invokespecial #13 // Method '<init>':(Ljava/lang/String;D)V 26: return public Employee(); Code: 0: aload_0 1: invokespecial #1 // Method java/lang/Object.'<init>':()V 4: aload_0 5: ldc #2 // String 7: putfield #3 // Field name:Ljava/lang/String; 10: aload_0 11: getstatic #4 // Field nextId:I 14: putfield #5 // Field id:I 17: getstatic #4 // Field nextId:I 20: iconst_1 21: iadd 22: putstatic #4 // Field nextId:I 25: return

標簽: java
相關文章:
主站蜘蛛池模板: 日韩免费一级毛片欧美一级日韩片 | 亚洲欧美色综合一区二区在线 | 欧美精品亚洲精品 | 日本99视频 | 久久日本精品国产精品白 | 欧美日本韩国一区二区 | 国产成人夜间影院在线观看 | 国产精品不卡在线观看 | 善良的翁熄日本2 | 日本高清色视频在线观看免费 | 欧美黄区| 久久国产免费福利永久 | 中国xxxx视频播放免费 | 国产在线精品视频 | 国产精品爱久久久久久久 | 九九精| 国产99精品一区二区三区免费 | 亚洲第一在线播放 | 婷婷综合在线观看丁香 | 国产精品爱啪在线线免费观看 | 真实国产乱子伦对白视频37p | 清纯唯美亚洲综合五月天 | www.黄色在线 | 三级黄色免费 | 中文字幕 日韩在线 | 国产精品性视频免费播放 | 国产精品久久久久久久久久久搜索 | 免费日本一区 | 91黄色片 | 久久一区二区三区不卡 | 国产成本人三级在线观看网站 | 中国hd高清╳xxx | 视频在线观看一区 | 成人免播放器午夜视频 | 综合久久 五十路 二区 | 日本在线不卡免费视频一区 | 91视频免费网址 | 99久久精品费精品国产一区二区 | 久免费视频| 国产区成人精品视频 | 成人久久伊人精品伊人 |