Java類的初始化順序

来源:https://www.cnblogs.com/xiaohang123/archive/2019/12/10/12019542.html

Java類的初始化順序 多說無益,以下是本人親自試驗的代碼,一目瞭然: 1 package test1; 2 3 public class Test { 4 public static void main(String[] argc) { 5 new Child(); 6 System.out.pr ...


Java類的初始化順序

多說無益,以下是本人親自試驗的代碼,一目瞭然:

 1 package test1;
 2 
 3 public class Test {
 4     public static void main(String[] argc) {
 5         new Child();
 6         System.out.println("================================");
 7         new Child();
 8     }
 9 }
10 
11 class Child extends Parent{
12     private static String staticField = getStaticField();
13     static {
14         System.out.println("child靜態方法塊初始化");
15     }
16     private String field = getField();
17     {
18         System.out.println("child普通方法塊初始化");
19     }
20 
21     public Child() {
22         System.out.println("child構造函數初始化");
23     }
24 
25     public static String getStaticField() {
26         System.out.println("child靜態屬性初始化");
27         return "staticField";
28     }
29 
30     public static String getField() {
31         System.out.println("child普通屬性初始化");
32         return "field";
33     }
34 }
35 
36 class Parent {
37     private static String staticField = getStaticField();
38     static {
39         System.out.println("parent靜態方法塊初始化");
40     }
41     private String field = getField();
42     {
43         System.out.println("parent普通方法塊初始化");
44     }
45 
46     public Parent() {
47         System.out.println("parent構造函數初始化");
48     }
49 
50     public static String getStaticField() {
51         System.out.println("parent靜態屬性初始化");
52         return "staticField";
53     }
54 
55     public static String getField() {
56         System.out.println("parent普通屬性初始化");
57         return "field";
58     }
59 }

列印結果:

 1 parent靜態屬性初始化
 2 parent靜態方法塊初始化
 3 child靜態屬性初始化
 4 child靜態方法塊初始化
 5 parent普通屬性初始化
 6 parent普通方法塊初始化
 7 parent構造函數初始化
 8 child普通屬性初始化
 9 child普通方法塊初始化
10 child構造函數初始化
11 ================================
12 parent普通屬性初始化
13 parent普通方法塊初始化
14 parent構造函數初始化
15 child普通屬性初始化
16 child普通方法塊初始化
17 child構造函數初始化

 經過替換靜態屬性和靜態初始化塊的聲明順序發現靜態屬性和靜態初始化塊的初始化順序和聲明順序有關,同理,普通屬性和普通初始化塊的的初始化順序和聲明順序有關

總結:

  1. 當某一個類滿足初始化的條件時(以後的博客中會有總結),先會初始化父類(從頂級父類Object依次向下初始化),然後初始化子類;
  2. 初始化該類時,最先初始化靜態屬性和靜態初始化塊(和聲明順序有關),從頂級父類Object依次向下初始化;
  3. 通過new創建對象時,先初始化普通屬性和普通初始化塊(和聲明順序有關),再調用構造方法,同樣從頂級父類Object依次向下執行;
  4. 靜態屬性和靜態初始化塊只會初始化一次,類初始化後,再次通過new創建對象,只會重覆執行第三步。

  接下來引用Java編程思想的一句話:

在類的內部,變數定義的先後順序決定了初始化的順序,即使變數定義散佈於方法定義之間,它們仍舊會在任何方法(包括構造器)被調用之前得到初始化。

參考:Java編程思想中文版第四版第5章

  及 https://www.cnblogs.com/fly-piglet/p/8766226.html


您的分享是我們最大的動力!

更多相關文章
  • 引入CDN,算好需要合併的單元格。 <!DOCTYPE html> <html> <head> <!-- 移動設備 --> <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no"> < ...
  • 簡介 適配器模式(Adapter Pattern)是作為兩個不相容的介面之間的橋梁。這種類型的設計模式屬於結構型模式,它結合了兩個獨立介面的功能。這種模式涉及到一個單一的類,該類負責加入獨立的或不相容的介面功能。 功能展示 場景模擬 外國出差,給筆記本充電,德國有一套德國標準,中國有一套中國標準(國 ...
  • 簡介 裝飾器模式(Decorator Pattern)允許向一個現有的對象添加新的功能,同時又不改變其結構。這種類型的設計模式屬於結構型模式,它是作為現有的類的一個包裝。 這種模式創建了一個裝飾類,用來包裝原有的類,併在保持類方法簽名完整性的前提下,提供了額外的功能。我們通過下麵的實例來演示裝飾器模 ...
  • '''''' ''' 1、簡述面向對象三大特性並用示例解釋說明?【背寫】 1、封裝 狹義的封裝:把一組屬性封裝到一個對象,創建對象的時候 廣義的封裝:代碼塊,函數、對象、類、模塊-py文件都是封裝 把封裝後的對象看成一個黑盒子,只需要關註輸入和輸出,不必關註黑盒子內部的實現 2、繼承 1、避免代..... ...
  • 今天在使用go與php的AES加解密交互中,一直有個問題那就是在go中加密後,在php端始終都是無法解密,經過排查最後發現是加密key長度引起的問題, 這裡簡單記錄下。 go的AES使用的是第三方的庫, "openssl" ,因為用的匆忙,沒註意看文檔,所以就直接弄了示例代碼,才發現和php端無法解 ...
  • 此模式通過一個模板方法來定義程式的框架或演算法,通常模板方法定義在基類中,即原始的模板,然後子類就可以根據不同的需要實現或重寫模板方法中的某些演算法步驟或者框架的某部分,最後達到使用相同模板實現不同功能的效果。 核心思想: 使用一個模板方法定義好總的演算法框架。 子類中根據需要重新定義某些操作,但是不能修 ...
  • 有個需求,從某個介面下載的一個zip壓縮包,往裡面添加一個說明文件。搜索了一下,沒有找到往zip直接添加文件的方法,最終解決方法是先解壓、再壓縮。具體過程如下: ...
  • 屬性 語法格式:修飾符 類型 屬性名 = 初值; 說明: 修飾符:public、protected、private:用於表示成員變數的訪問許可權。static:表示該成員變數為類變數,也稱為靜態變數。final:表示將該成員變數聲明為常量,其值無法更改。 類型:表示變數的類型。 屬性名:表示變數名稱。 ...
一周排行
  • " 返回《C 併發編程》" "1. 概念介紹" "2. 非同步編程" "2.1. async運行過程" "2.2. async運行中同步上下文簡介" "2.3. 創建Task實例" "2.4. 捕獲非同步異常類型" "3. 並行編程" "3.1. Parallel" "3.2. 異常處理" "3.3. ...
  • 我們先看看兩個特效,感受一下,有沒有學習的動力? 核心API:Texture2D.SetPixel(int x, int y, Color color),Texture2D.Apply() 實現原理:對象池 思路: 第一幀繪製前:遍歷瓦片上所有活著的粒子對象並且進行數據操作(或運動,死亡),發生運動 ...
  • 原來的導出方式比較適用於比較簡單的導出,每一條數據在一行,數據列雖然自定義程度比較高,如果要一條數據對應多行就做不到了,於是就想支持根據模板導出,在 1.8.0 版本中引入了根據模板導出的功能 ...
  • 創建一個bat腳本, 裡面寫上: reg delete HKEY_CURRENT_USER\Software\JetBrains\dotMemory /freg delete HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Ex ...
  • Linux下有vsyscall來優化一些例如time(NULL), gettimeofday這種調用的消耗; 但是Windows下, 沒有類似的東西, 但是思路還是有的 1. 程式啟動的時候, 獲取一下準確的時間戳 2. 然後每次需要獲取時間的時候, 獲取一下流逝的時間, 可以通過獲取CPU的tic ...
  • 當用戶按下鍵盤上的一個鍵時,就會發生一系列事件。下表根據他們的發生順序列出了這些事件: 表 所有元素的鍵盤事件(按順序) 鍵盤處理永遠不會像上面看到的這麼簡單。一些控制項可能會掛起這些事件中的某些事件,從而可執行自己更特殊的鍵盤處理。最明顯的例子是TextBox控制項,它掛起了TextInput事件。對 ...
  • static void LocalMethod() { Cube(100); void Cube(int x) => Console.WriteLine($"The cube of {x} is {x * x * x}"); } static void GoToDemo() { int i = 1; ...
  • 滑鼠事件執行幾個關聯的任務。當滑鼠移到某個元素上時,可通過最基本的滑鼠事件進行響應。這些事件是MouseEnter(當滑鼠指針移到元素上時引發該事件)和MouseLeave(當滑鼠指針離開元素時引發該事件)。這兩個事件都是直接事件,這意味著他們不使用冒泡和隧道過程,而是源自一個元素並且只被該元素引發 ...
  • 反射這個詞聽起來就很牛逼是吧? 嗯的確,反射是比較高級的特性,只有語言基礎很扎實的Dev們才應該使用它。 搞點反射,可以提高程式的靈活性、可擴展性、耦合度。 反射這東西,是為了動態地運行時載入,相比於靜態代碼。編譯的時候就是板上釘釘了。 就是說,如果你的程式需要在運行時搞一些晚綁定,動態載入或檢查對 ...
  • 眾所周知,微服務架構是由一眾微服務組成,項目中調用其他微服務介面更是常見的操作。為了便於調用外部介面,我們的常用思路一般都是封裝一個外部介面的客戶端,使用時候直接調用相應的方法。webservice或WCF的做法就是引用服務,自動生成客戶端。在webapi2.0里,我們都會手動封裝一個靜態類。那麼在 ...
x