九、 Java程式初始化的順序(二)

来源:https://www.cnblogs.com/Jeffding/archive/2018/08/17/9490510.html
-Advertisement-
Play Games

之前的一篇博客里我寫了關於在一個類中的程式初始化順序,但是在Java的面向對象里,類之間還存在著繼承的關係。所以關於程式的初始化順序,我們可以再細劃分為:父類靜態變數,父類的靜態代碼塊,父類構造器,父類非靜態變數,父類非靜態代碼塊,子類靜態變數,子類靜態代碼塊,子類構造器,子類非靜態成員變數和子類非 ...


  之前的一篇博客里我寫了關於在一個類中的程式初始化順序,但是在Java的面向對象里,類之間還存在著繼承的關係。所以關於程式的初始化順序,我們可以再細劃分為:父類靜態變數,父類的靜態代碼塊,父類構造器,父類非靜態變數,父類非靜態代碼塊,子類靜態變數,子類靜態代碼塊,子類構造器,子類非靜態成員變數和子類非靜態代碼塊。
本篇博客我們討論的就是關於程式初始化的過程中,上述的成員在初始化載入先後順序。

  在此前我們討論得出的結論:在一個類中,Java程式載入的順序是:靜態變數-->靜態代碼塊-->非靜態變數-->非靜態代碼塊-->構造器.

  父類的代碼:
public class SuperClass {
    //父類與子類都在一個包中,這裡我們就使用default修飾符
    //這是一個父類的靜態變數,此時還是初始化的預設值null
    static String superStaticVariale;

    // 靜態代碼塊,給String賦值
    static {
        superStaticVariale = "父類靜態代碼塊賦值成功";
        System.out.println("此時運行的是父類的靜態代碼塊:"+superStaticVariale);
    }

    // 無參構造,覆蓋靜態代碼塊的值
    SuperClass(){
        superStaticVariale = "父類構造器賦值成功";
        System.out.println("此時運行的是父類的構造器:"+superStaticVariale);
    }

    //定義一個非靜態變數
    String superVariale;

    // 定義一個非靜態代碼塊
    {
        superVariale = "父類非靜態代碼塊賦值";
        System.out.println("此時運行的是父類的非靜態代碼塊:"+superVariale);
    }
}
  子類的代碼:
public class SubClass extends SuperClass{
    static String subStaticVariale;

    // 靜態代碼塊,給String賦值
    static {
        subStaticVariale = "子類靜態代碼塊賦值成功";
        System.out.println("此時運行的是子類的靜態代碼塊:"+subStaticVariale);
    }

    // 無參構造,覆蓋靜態代碼塊的值
    SubClass(){
        superStaticVariale = "子類構造器賦值成功";
        System.out.println("此時運行的是子類的構造器:"+superStaticVariale);
    }

    //定義一個非靜態變數
    String subVariale;

    // 定義一個非靜態代碼塊
    {
        subVariale = "子類非靜態代碼塊賦值";
        System.out.println("此時運行的是子類非靜態代碼塊:"+subVariale);
    }
}
  測試代碼:
public class Main {

    public static void main(String[] args) {
        SubClass s = new SubClass();
    }
}
  運行結果:
```
此時運行的是父類的靜態代碼塊:父類靜態代碼塊賦值成功
此時運行的是子類的靜態代碼塊:子類靜態代碼塊賦值成功
此時運行的是父類的非靜態代碼塊:父類非靜態代碼塊賦值
此時運行的是父類的構造器:父類構造器賦值成功
此時運行的是子類非靜態代碼塊:子類非靜態代碼塊賦值
此時運行的是子類的構造器:子類構造器賦值成功
```
  很顯然,在繼承關係中,代碼的載入順序是:父類的靜態變數-->父類的靜態代碼塊-->子類靜態變數-->子類的靜態代碼塊-->父類非靜態變數-->父類的非靜態代碼塊-->父類的構造器-->子類非靜態變數-->子類非靜態代碼塊-->子類構造器

  進一步測試:
public class Main {

    public static void main(String[] args) {
        SubClass s = new SubClass();
        SubClass s1 = new SubClass();
        SubClass s2 = new SubClass();
    }
}
運行結果:
```
此時運行的是父類的靜態代碼塊:父類靜態代碼塊賦值成功
此時運行的是子類的靜態代碼塊:子類靜態代碼塊賦值成功
此時運行的是父類的非靜態代碼塊:父類非靜態代碼塊賦值
此時運行的是父類的構造器:父類構造器賦值成功
此時運行的是子類非靜態代碼塊:子類非靜態代碼塊賦值
此時運行的是子類的構造器:子類構造器賦值成功
此時運行的是父類的非靜態代碼塊:父類非靜態代碼塊賦值
此時運行的是父類的構造器:父類構造器賦值成功
此時運行的是子類非靜態代碼塊:子類非靜態代碼塊賦值
此時運行的是子類的構造器:子類構造器賦值成功
此時運行的是父類的非靜態代碼塊:父類非靜態代碼塊賦值
此時運行的是父類的構造器:父類構造器賦值成功
此時運行的是子類非靜態代碼塊:子類非靜態代碼塊賦值
此時運行的是子類的構造器:子類構造器賦值成功
```
  得出結論:
  父類與子類的靜態代碼都只執行一次,然後非靜態代碼塊與構造器是組合出現的。

  簡化一下代碼:
public class Main {

    public static void main(String[] args) {
   C c= new C();
    }
}

class A{
    A(){
        System.out.println("A的無參構造器");
    }
}

class B extends A{
//    B(int a){
    B(){
        System.out.println("B的無參構造器");
    }
}

class C extends B{
    C(){
        System.out.println("C的無參構造器");
    }
}
  運行結果:
```text
A的無參構造器
B的無參構造器
C的無參構造器
```
  調用C的構造器生成C的實例對象會從最上級的父類的無參構造器開始逐層調用,那麼我們的類都繼承了一個超級父類Object,也就是在我們最初的錯誤代碼中,我們調用Student的無參構造創建一個對象時,首先會調用這個對象的父類Object的無參構造器,
class Student{
   String name;
   
   {
      name = "老大";
   }
   
   Student(){
       this(name);//這樣會報錯
      super();
      System.out.println("題目要求寫一個無參的構造器");
   }
   
   Student(String name){
      this.name = name;
      System.out.println(name);
   }
   
}
  子類實例化預設調用父類的無參構造器,也就是如上this調用在super()之前(實際中這兩者不會同時出現),name此時是非靜態屬性,此時會報錯錯誤: 無法在調用超類型構造器之前引用name。
class Student{
   static String name;
   
   {
      name = "老大";
   }
   
   Student(){
       this(name);
      System.out.println("題目要求寫一個無參的構造器");
   }
   
   Student(String name){
      this.name = name;
      System.out.println(name);
   }
   
}
  當name是靜態屬性時,代碼塊是非靜態時,編譯通過,調用子類的無參構造器時this(name),輸出結果是:
```text
null
題目要求寫一個無參的構造器
```
此時的this()調用實參構造並沒有賦值成功。
class Student{
   static String name;
   
   static{
      name = "老大";
   }
   
   Student(){
       this(name);
      System.out.println("題目要求寫一個無參的構造器");
   }
   
   Student(String name){
      this.name = name;
      System.out.println(name);
   }
}
  此時運行結果:
```text
老大
題目要求寫一個無參的構造器
```
  這樣賦值成功。由此證明我們的結論是正確的,this()是在子類父類構造器之前進行的操作super(),當子類代碼塊是非靜態時,子類非靜態代碼塊會在執行父類構造器之後執行,所以this(name)時name還沒有被賦值,所以列印是null。

  結論:
  1. 一個類中可以在無參構造器中調用此類的有參構造器(順序反過來);
  2. 在執行子類的無參構造器時會預設調用最高級父類無參構造,並逐級調用直至子類的無參構造;
  3. Java程式的載入順為父類的靜態變數-->父類的靜態代碼塊-->子類靜態變數-->子類的靜態代碼塊-->父類非靜態變數-->父類的非靜態代碼塊-->父類的構造器-->子類非靜態變數-->子類非靜態代碼塊-->子類構造器,且靜態變數或代碼塊無論構造器調用多少次,他只會執行一次,後面再調用構造器則會執行非靜態屬性及代碼塊構造器。

  最後關於為什麼子類會調用父類的構造器,這個從設計著的角度來看是為了給從父類繼承的屬性初始化,子類需要知道父類是如何給屬性初始化的。

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

-Advertisement-
Play Games
更多相關文章
  • 1. jQuery函數的基本語法: $(document).ready(function(){ //代碼塊; }) 2.window.onload()和$(document).ready()的區分: 1)window.onload()它的執行時機是必須等待網頁中所有的內容載入完畢(包括圖片)才能執行 ...
  • 本文 基於 JQuery 擴展了一些 JS日期函數,包括: > 字元串 轉 Date 對象 萬能函數(性能僅 10W次/s,函數有路徑優化,字元串越詭異 耗時越長) > Date 轉 字元串 格式化 > 兩個 Date 的差值 (返回的結果類似 C# TimeSpan 對象) ...
  • var u = navigator.userAgent; var isAndroid = u.indexOf('Android') > -1 || u.indexOf('Adr') > -1; //android終端 var isiOS = !!u.match(/\(i[^;]+;( U;)? CP... ...
  • 前言 周末自己在家閑著沒事,刷著微信,玩著手機,發現自己的微信頭像該換了,就去網上找了一下頭像,看著圖片,自己就想著作為一個碼農,可以把這些圖片都爬取下來做成一個微信小程式,說乾就乾,瞭解一下基本都知道怎麼做了,整理分享一波給大家。 目錄 前言 周末自己在家閑著沒事,刷著微信,玩著手機,發現自己的微 ...
  • 1、首先我們要獲取當前點擊的對象,然後得到點擊事件, 判斷他的狀態如果是checked的話就把該第二行的選中, 否則就取消選中。 2、當第二列功能小項沒有全部選中時,該行第一列的覆選款也要取消選中。 ...
  • @*以下 是測試html 顏色選擇器的*@ @*<a href="#" mce_href="#" onclick="initColorPicker('demo'); return false" id="demo" style="position:absolute;left:200px">顏色選擇</ ...
  • 題面描述 試題描述 小a和小b玩一個游戲,有n張卡牌,每張上面有兩個正整數x,y。 取一張牌時,個人積分增加x,團隊積分增加y。 求小a,小b各取若幹張牌,使得他們的個人積分相等。 輸入 第一行一個整數n。接下來n行,每行兩個整數x,y,用空格隔開。 輸出 一行一個整數,表示小a的積分和小b的積分相 ...
  • c/c++賦值函數(重載=號運算符) 首先c++里的各種運算符都是用函數實現的,比如=,就等號函數。 所以當用=給一個對象賦值的時候,實際調用的是=號所對應的=號函數。 分析下麵的代碼 c++ include using namespace std; class Test{ public: expl ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...