B2B网站_日本理论_B2B免费发布信息网站_日本看片网站_B2B企业贸易平台 -日本看片网站- 企资网

二維碼
企資網

掃一掃關注

當前位置: 首頁 » 企業資訊 » 資訊 » 正文

一題搞定static關鍵字

放大字體  縮小字體 發布日期:2021-09-05 06:33:58    作者:企資小編    瀏覽次數:82
導讀

基礎不牢,地動山搖。大家好,我是課代表。可以關注我的公眾號:Java課代表,原創實戰干貨首發地兒,等你呦。開篇一道題,考察代碼執行順序:public class Parent {static {System.out.println("Parent static i

基礎不牢,地動山搖。

大家好,我是課代表。可以關注我的公眾號:Java課代表,原創實戰干貨首發地兒,等你呦。

開篇一道題,考察代碼執行順序:

public class Parent {    static {        System.out.println("Parent static initial block");    }    {        System.out.println("Parent initial block");    }    public Parent() {        System.out.println("Parent constructor block");    }}public class Child extends Parent {    static {        System.out.println("Child static initial block");    }    {        System.out.println("Child initial block");    }        private Hobby hobby = new Hobby();    public Child() {        System.out.println("Child constructor block");    }}public class Hobby {    static{        System.out.println("Hobby static initial block");    }    public Hobby() {        System.out.println("hobby constructor block");    }}

當執行new Child()時,上述代碼輸出什么?

相信有不少同學遇到過這類問題,可能查過資料之后接著就忘了,再次遇到還是答不對。接下來課代表通過4個步驟,帶大家拆解一下這段代碼的執行順序,并借此總結規律。

1.編譯器優化了啥?

下面兩段代碼對比一下編譯前后的變化:

編譯前的Child.java

public class Child extends Parent {    static {        System.out.println("Child static initial block");    }    {        System.out.println("Child initial block");    }        private Hobby hobby = new Hobby();        public Child() {        System.out.println("Child constructor block");    }}

編譯后的Child.class

public class Child extends Parent {    private Hobby hobby;    public Child() {        System.out.println("Child initial block");        this.hobby = new Hobby();        System.out.println("Child constructor block");    }    static {        System.out.println("Child static initial block");    }}

通過對比可以看到,編譯器把初始化塊和實例字段的賦值操作,移動到了構造函數代碼之前,并且保留了相關代碼的先后順序。事實上,如果構造函數有多個,初始化代碼也會被復制多份移動過去。

據此可以得出第一條優先級順序:

  • 初始化代碼 > 構造函數代碼

    2.static 有啥作用?

    類的加載過程可粗略分為三個階段:加載 -> 鏈接 -> 初始化

    初始化階段可被8種情況觸發:

    1. 使用 new 關鍵字實例化對象的時候
    2. 讀取或設置一個類型的靜態字段(常量除外)
    3. 調用一個類型的靜態方法
    4. 使用反射調用類的時候
    5. 當初始化類的時候,如果發現父類還沒有進行過初始化,則先觸發其父類初始化
    6. 虛擬機啟動時,會先初始化主類(包含main()方法的那個類)
    7. 當初次調用 MethodHandle 實例時,初始化該 MethodHandle 指向的方法所在的類。
    8. 如果接口中定義了默認方法(default 修飾的接口方法),該接口的實現類發生了初始化,則該接口要在其之前被初始化

    其中的2,3條目是被static代碼觸發的。

    其實初始化階段就是執行類構造器<clinit> 方法的過程,這個方法是編譯器自動生成的,里面收集了static修飾的所有類變量的賦值動作和靜態語句塊(static{} 塊),并且保留這些代碼出現的先后順序。

    根據條目5,JVM 會保證在子類的<clinit>方法執行前,父類的<clinit>方法已經執行完畢。

    小結一下:訪問類變量或靜態方法,會觸發類的初始化,而類的初始化就是執行<clinit>,也就是執行 static 修飾的賦值動作和static{}塊,并且 JVM 保證先執行父類初始化,再執行子類初始化。

    由此得出第二條優先級順序:

  • 父類的static代碼 > 子類的static代碼

    3.static 代碼只執行一次

    我們都知道,static代碼(靜態方法除外)只執行一次。

    你有沒有想過,這個機制是如何保證的呢?

    答案是:雙親委派模型。

    JDK8 及之前的雙親委派模型是:

    應用程序類加載器 → 擴展類加載器 → 啟動類加載器

    平時開發中寫的類,默認都是由 應用程序類加載器加載,它會委派給其父類:擴展類加載器。而擴展類加載器又會委派給其父類:啟動類加載器。只有當父類加載器反饋無法完成這個加載請求時,子加載器才會嘗試自己去完成加載,這個過程就是雙親委派。三者的父子關系并不是通過繼承,而是通過組合模式實現的。

    該過程的實現也很簡單,下面展示關鍵實現代碼:

    protected Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException{    // 首先檢查該類是否被加載過    // 如果加載過,直接返回該類    Class<?> c = findLoadedClass(name);    if (c == null) {        try {            if (parent != null) {                c = parent.loadClass(name, false);            } else {                c = findBootstrapClassOrNull(name);            }        } catch (ClassNotFoundException e) {            // 如果父類拋出ClassNotFoundException            // 說明父類無法完成加載請求        }        if (c == null) {            // 如果父類無法加載,轉由子類加載            c = findClass(name);        }    }    if (resolve) {        resolveClass(c);    }    return c;}

    結合注釋相信大家很容易看懂。

    由雙親委派的代碼可知,同一個類加載器下,一個類只能被加載一次,也就限定了它只能被初始化一次。所以類中的 static代碼(靜態方法除外)只在類初始化時執行一次

    4. <init>和<clinit>

    前面已經介紹了編譯器自動生成的類構造器:<clinit>方法,它會收集static修飾的所有類變量的賦值動作和靜態語句塊(static{} 塊)并保留代碼的出現順序,它會在類初始化時執行

    相應的,編譯器還會生成一個<init>方法,它會收集實例字段的賦值動作、初始化語句塊({}塊)和構造器(Constructor)中的代碼,并保留代碼的出現順序,它會在 new 指令之后接著執行

    所以,當我們new 一個類時,如果JVM未加載該類,則先對其進行初始化,再進行實例化。

    至此,第三條優先級規則也就呼之欲出了:

  • 靜態代碼(static{}塊、靜態字段賦值語句) > 初始化代碼({}塊、實例字段賦值語句)

    5. 規律實踐

    將前文的三條規則合并,總結出如下兩條:

    1.靜態代碼(static{}塊、靜態字段賦值語句) > 初始化代碼({}塊、實例字段賦值語句) > 構造函數代碼

    2.父類的static代碼 > 子類的static代碼

    根據前文總結,初始化代碼和構造函數代碼被編譯器收集到了<init>中,靜態代碼被收集到了<clinit>中,所以再次對上述規律做合并:

    父類<clinit> > 子類<clinit> > 父類 <init> > 子類 <init>

    對應到開篇的問題,我們來實踐一下:

    當執行new Child()時,new關鍵字觸發了 Child 類的初始化 ,JVM 發現其有父類,則先初始化 Parent 類,開始執行Parent類的<clinit>方法,然后執行 Child 類的<clinit>方法(還記得<clinit>里面收集了什么嗎?)。

    然后開始實例化 一個Child類的對象,此時準備執行 Child 的<init>方法,發現它有父類,優先執行父類的<init>方法,然后再執行子類的<init>(還記得<init>里面收集了什么嗎?)。

    相信看到這里,各位心中已經對開篇的問題有答案了,不妨先手寫一下輸出順序,然后寫代碼親自驗證一下。

    結束語

    平時開發中經常用到static,每次寫的時候,心里總會打兩個問號,我為什么要用static?不用行不行?這正應了開篇的第一句話:

    基礎不牢,地動山搖

    通過本文可以看出,static的應用遠遠不止類變量,靜態方法那么簡單。在經典的單例模式中,你將看到static的各種用法,下一篇就寫如何花式編寫單例模式。


    原創碼字不容易,歡迎關注點贊和分享。

  •  
    (文/企資小編)
    免責聲明
    本文僅代表作發布者:企資小編個人觀點,本站未對其內容進行核實,請讀者僅做參考,如若文中涉及有違公德、觸犯法律的內容,一經發現,立即刪除,需自行承擔相應責任。涉及到版權或其他問題,請及時聯系我們刪除處理郵件:weilaitui@qq.com。
     

    Copyright ? 2016 - 2025 - 企資網 48903.COM All Rights Reserved 粵公網安備 44030702000589號

    粵ICP備16078936號

    微信

    關注
    微信

    微信二維碼

    WAP二維碼

    客服

    聯系
    客服

    聯系客服:

    在線QQ: 303377504

    客服電話: 020-82301567

    E_mail郵箱: weilaitui@qq.com

    微信公眾號: weishitui

    客服001 客服002 客服003

    工作時間:

    周一至周五: 09:00 - 18:00

    反饋

    用戶
    反饋

    主站蜘蛛池模板: 双舌接地线-PC68数字式高阻计-ZC36|苏海百科 | 深圳侦探联系方式_深圳小三调查取证公司_深圳小三分离机构 | 光照全温振荡器(智能型)-恒隆仪器| 真空包装机-诸城市坤泰食品机械有限公司 | lcd条形屏-液晶长条屏-户外广告屏-条形智能显示屏-深圳市条形智能电子有限公司 | 翅片管散热器价格_钢制暖气片报价_钢制板式散热器厂家「河北冀春暖气片有限公司」 | 防爆电机生产厂家,YBK3电动机,YBX3系列防爆电机,YBX4节防爆电机--河南省南洋防爆电机有限公司 | 成都热收缩包装机_袖口式膜包机_高速塑封机价格_全自动封切机器_大型套膜机厂家 | 5nd音乐网|最新流行歌曲|MP3歌曲免费下载|好听的歌|音乐下载 免费听mp3音乐 | 湖南长沙商标注册专利申请,长沙公司注册代理记账首选美创! | 电动葫芦|手拉葫芦|环链电动葫芦|微型电动葫芦-北京市凌鹰起重机械有限公司 | 电动车头盔厂家_赠品头盔_安全帽批发_山东摩托车头盔—临沂承福头盔 | 裹包机|裹膜机|缠膜机|绕膜机-上海晏陵智能设备有限公司 | 西门子伺服控制器维修-伺服驱动放大器-828D数控机床维修-上海涌迪 | 南京技嘉环保科技有限公司-杀菌除臭剂|污水|垃圾|厕所|橡胶厂|化工厂|铸造厂除臭剂 | 艾默生变频器,艾默生ct,变频器,ct驱动器,广州艾默生变频器,供水专用变频器,风机变频器,电梯变频器,艾默生变频器代理-广州市盟雄贸易有限公司官方网站-艾默生变频器应用解决方案服务商 | 润滑油加盟_润滑油厂家_润滑油品牌-深圳市沃丹润滑科技有限公司 琉璃瓦-琉璃瓦厂家-安徽盛阳新型建材科技有限公司 | 美国PARKER齿轮泵,美国PARKER柱塞泵,美国PARKER叶片泵,美国PARKER电磁阀,美国PARKER比例阀-上海维特锐实业发展有限公司二部 | 冲锋衣滑雪服厂家-冲锋衣定制工厂-滑雪服加工厂-广东睿牛户外(S-GERT) | 厂房出租-厂房规划-食品技术-厂房设计-厂房装修-建筑施工-设备供应-设备求购-龙爪豆食品行业平台 | 蜂蜜瓶-玻璃瓶-玻璃瓶厂-玻璃瓶生产厂家-徐州贵邦玻璃制品有限公司 | 合金ICP光谱仪(磁性材料,工业废水)-百科 | 舞台木地板厂家_体育运动木地板_室内篮球馆木地板_实木运动地板厂家_欧氏篮球地板推荐 | 不锈钢列管式冷凝器,换热器厂家-无锡飞尔诺环境工程有限公司 | 【化妆品备案】进口化妆品备案流程-深圳美尚美化妆品有限公司 | 圆形振动筛_圆筛_旋振筛_三次元振动筛-河南新乡德诚生产厂家 | 模具硅橡胶,人体硅胶,移印硅胶浆厂家-宏图硅胶科技 | 甲级防雷检测仪-乙级防雷检测仪厂家-上海胜绪电气有限公司 | 青州搬家公司电话_青州搬家公司哪家好「鸿喜」青州搬家 | 河南包装袋厂家_河南真空袋批发价格_河南服装袋定制-恒源达包装制品 | 上海logo设计 | 大通天成企业资质代办_承装修试电力设施许可证_增值电信业务经营许可证_无人机运营合格证_广播电视节目制作许可证 | 交变/复合盐雾试验箱-高低温冲击试验箱_安奈设备产品供应杭州/江苏南京/安徽马鞍山合肥等全国各地 | 石家庄小程序开发_小程序开发公司_APP开发_网站制作-石家庄乘航网络科技有限公司 | 上海皓越真空设备有限公司官网-真空炉-真空热压烧结炉-sps放电等离子烧结炉 | 耐火浇注料价格-高强高铝-刚玉碳化硅耐磨浇注料厂家【直销】 | 山东商品混凝土搅拌楼-环保型搅拌站-拌合站-分体仓-搅拌机厂家-天宇 | 苏州伊诺尔拆除公司_专业酒店厂房拆除_商场学校拆除_办公楼房屋拆除_家工装拆除拆旧 | 辊道窑炉,辊道窑炉厂家-山东艾希尔| 注塑机-压铸机-塑料注塑机-卧式注塑机-高速注塑机-单缸注塑机厂家-广东联升精密智能装备科技有限公司 | 【黄页88网】-B2B电子商务平台,b2b平台免费发布信息网 |