c++模板类静态成员变量_一文讲透父子类中静态变量,成员变量初始化顺序原理...

推荐:

  • 从面试到入职到离职,我在B站工作的30天时光
  • 爱了爱了,Alibaba顶级MySQL调优手册到手,加薪妥了
  • 爱了爱了,Spring Cloud Alibaba内部微服务架构笔记真的太牛了

本文主要是想分析一下静态变量和成员变量的初始化顺序,以及如果存在父类和子类关系时,这些变量的顺序又是如何呢?本文将一一进行分析。

7398e1beeec972ec9c362f14f141160b.png

类加载初始化和实例对象初始化

首先我们要区分一个概念,那就是变量初始化也可以分为两大类, 一类是Java虚拟机中的类加载机制内有一个初始化,而我们实例化一个对象(即通过new关键字)时也有一个对象实例的初始化。

首先我们要知道,一个类理论上不考虑内存影响的话可以被实例化无数次,所以我们想要实例化一个对象,那么就必须要有这个类,所有类加载一定在对象实例化前面。

而静态变量是在类加载阶段就会进行初始化的(如果想详细了解Java虚拟机的类加载机制,可以点击这里),故而如果在同一个类中,那么静态变量的初始化一定在成员变量的初始化前面

静态变量和成员变量区别

我们先来看下面两句简单的代码:

package com.zwx.coreJava;public class InitVariable {    public static int m = 1;//静态变量(类变量)    private int n = 2;//成员变量(实例变量)    public static void main(String[] args) {      System.out.println(InitVariable.m);//直接通过类访问变量        InitVariable initVariable = new InitVariable();//创建一个实例对象        System.out.println(initVariable.n);//需要通过实例对象才能访问成员变量    }}

其中m就是静态变量,也称之为类变量,类加载之后,存储在方法区内,可以直接通过:类名.变量名进行调用。如:对象.m。

其中n就是成员变量,也称之为实例变量,属于对象实例,需要通过new一个对象之后才能引用。如:new 对象().n。成员变量的生命周期是和实例对象一致的,当实例对象被回收之后,对应的成员变量也消失了。

变量加载顺序

加下来就让我们一起结合例子验证一下变量的加载顺序

静态变量和成员变量加载顺序

首先我们建立一个父类,然后仅以父类为例子验证下静态变量和成员变量的加载顺序:

package com.zwx.coreJava;public class SuperInitVariable {    static String superStaticStr = "I'm Super Static Str1";    String superInstanceStr = "I'm Super Instance Str1";    static {        superStaticStr = "I'm Super Static Str2";    }    static {        superStaticStr = "I'm Super Static Str3";    }    public SuperInitVariable() {        superInstanceStr = "I'm Super Instance Str2";    }}

然后新建一个测试类:

package com.zwx.coreJava;public class TestVariableOrder {    public static void main(String[] args) {        System.out.println(SuperInitVariable.superStaticStr);        SuperInitVariable superInitVariable = new SuperInitVariable();        System.out.println(superInitVariable.superInstanceStr);    }}

输出结果:

I'm Super Static Str3I'm Super Instance Str2

很明显,第一句话还没有实例化对象就可以输出静态变量,所以静态变量优先级最高,静态代码块也是一样会加载,多个代码块之间按先后顺序加载。
第二句成员变量先初始化了变量,再执行了构造器。所以可以得出如下结论:

  • 1、静态变量优先级最先被初始化,多个静态代码块按代码先后顺序进行加载
  • 2、成员变量先赋值,再执行构造函数

父类和子类变量加载顺序

将上面的父类简单改造一下:

package com.zwx.coreJava;public class SuperInitVariable {    static String superStaticStr = "I'm Super Static Str1";    String superInstanceStr = "I'm Super Instance Str1";    static {        superStaticStr = "I'm Super Static Str2";        System.out.println("Super Static:" + superStaticStr);    }    public SuperInitVariable() {        superInstanceStr = "I'm Super Instance Str2";        System.out.println("Super Construct:" + superStaticStr);    }}

然后再新建一个子类,继承上面的SuperInitVariable:

package com.zwx.coreJava;public class SubInitVariable extends SuperInitVariable {    static String subStaticStr = "I'm Sub Static Str1";    String subInstanceStr = "I'm Sub Instance Str1";    static {        subStaticStr = "I'm Sub Static Str2";        System.out.println("Sub Static:" + subStaticStr);    }    public SubInitVariable() {        subInstanceStr = "I'm Sub Instance Str1";        System.out.println("Sub Construct:" + subInstanceStr);    }}

然后在测试类中执行以下语句:

package com.zwx.coreJava;public class TestVariableOrder {    public static void main(String[] args) {        System.out.println(SubInitVariable.subStaticStr);    }}

输出如下结果:

Super Static:I'm Super Static Str2Sub Static:I'm Sub Static Str2I'm Sub Static Str2

可以很明显看到加载顺序为:先加载父类静态变量,再加载子类静态变量
接下来再改造下测试类,我们去实例化一个SubInitVariable对象实例:

package com.zwx.coreJava;public class TestVariableOrder {    public static void main(String[] args) {        SubInitVariable subInitVariable = new SubInitVariable();    }}

输出结果如下:

Super Static:I'm Super Static Str2Sub Static:I'm Sub Static Str2Super Construct:I'm Super Static Str2Sub Construct:I'm Sub Instance Str1

可以很明显的得出如下结论:

  • 1、初始化父类静态变量。
  • 2、初始化子类静态变量。
  • 3、初始化父类成员变量。
  • 4、加载父类构造器。
  • 5、初始化子类成员变量。
  • 6、加载子类构造器。

或者说可以分的更细致一点,可以总结为如下:

  • 1、初始化父类静态变量。
  • 2、初始化父类静态代码块。
  • 3、初始化子类静态变量。
  • 4、初始化子类静态代码块。
  • 5、初始化父类成员变量。
  • 6、加载父类构造器。
  • 7、初始化子类成员变量。
  • 8、加载子类构造器。

引用类加载顺序

上面的例子中变量都是String类型的,那么假如变量是引用类型呢?又会怎么样?
我们再来看个例子。新建两个类,其中一个类引用另一个类:

package com.zwx.coreJava;public class ReferenceObj {    static String referenceStaticStr;    String referenceInstanceStr;    static {        referenceStaticStr = "I'm reference Static Str";        System.out.println(referenceStaticStr);    }    public ReferenceObj() {        referenceInstanceStr = "I'm reference Construct Str";        System.out.println(referenceInstanceStr);    }}
package com.zwx.coreJava;public class InitReferenceVariable {    static ReferenceObj staticObj;    ReferenceObj obj;    static {        System.out.println("static code:" + staticObj);    }    public InitReferenceVariable() {        System.out.println("construct code:" + obj);    }}

上面例子中看到,我们只是持有了另一个对象的引用,不做任何初始化动作,这时候是否会去加载引用类呢?我们新建一个测试类测试一下:

package com.zwx.coreJava;public class TestVariableOrder {    public static void main(String[] args) {        //测试引用变量和数组        System.out.println(InitReferenceVariable.staticObj);        InitReferenceVariable initReferenceVariable = new InitReferenceVariable();        System.out.println(initReferenceVariable.obj);    }}

输出如下结果:

static code:nullnullconstruct code:nullnull

可以看到,引用类ReferenceObj没有被实例化也没有被加载。
那么我们把上面的变量进行初始化修改一下:

static ReferenceObj staticObj = new ReferenceObj();ReferenceObj obj = new ReferenceObj();

再次运行测试类,输出如下结果:

I'm reference Static StrI'm reference Construct Strstatic code:com.zwx.coreJava.ReferenceObj@4b1210eecom.zwx.coreJava.ReferenceObj@4b1210eeI'm reference Construct Strconstruct code:com.zwx.coreJava.ReferenceObj@4d7e1886

可以看到:I’m reference Static Str只输出了1次,而I’m reference Construct Str输出了2次,这是因为同一个类只会被加载1次,但是我们new了两次,也就是实例化了2次,所以构造函数会执行2次。
根据这个结果我们可以得出如下结论:

  • 1、一个类假如只是引用另一个类而没有被实例化,那么不会触发引用类的类加载和实例化
  • 1、一个类假如引用另一个类并且实例化了引用类,那么会优先加载引用类和实例化引用类

数组引用类加载顺序

把类改造如下:

package com.zwx.coreJava;public class InitReferenceVariable {    static ReferenceObj[] arr = new ReferenceObj[10];}

然后在测试类中输出如下语句:

System.out.println("输出数组:" + InitReferenceVariable.arr);

输出结果如下:

输出数组:[Lcom.zwx.coreJava.ReferenceObj;@4b1210ee

可以看到只会输出这一句话,而不会触发ReferenceObj的类加载和初始化。
可以得出如下结论:
如果将引用类作为数组对象,那么创建数组的时候不会触发数组对象的类加载和初始化

总结

本文从父类,子类,引用类,以及数组引用类的角度全面分析了什么时候会触发一个类的加载,什么时候会触发一个类的实例化。而一个类在加载的时候就会初始化静态变量,一个类在实例化的时候就会触发成员变量的初始化。本文涉及到类加载时并没有特别深入的介绍,有兴趣的,可以看下本人下的JVM虚拟机系列2中讲述的类加载机制,相信看完之后会有更加深刻的理解和体会。

来源:http://002ii.cn/?T0vKxr

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/399307.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Cocos2D将v1.0的tileMap游戏转换到v3.4中一例(八)

大熊猫猪侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请告诉我,如果觉得不错请多多支持点赞.谢谢! hopy ;) 回到Xcode中,新建一个EndLayer类,继承于CCNode.因为我们得在其类方法中创建实例变量,所以我们得建立几个实例变量的属性.打开EndLayer.h文件,修改内…

转:(图文并茂)SQL Server 2005详细安装过程及配置

1、安装过程 注:在Windows7系统下安装SQL Server 2005时,可能会多次遇到提示兼容性问题的情况,此时不用理会,直接点击“运行程序”即可。首先解压下载到的SQL Server 2005压缩文件,如果还没有这个文件,可以…

2019年java安装步骤_win10的JAVA(jdk)2020年最新版安装教程心得

想学安卓反向解析,前期工作就把我折腾得够呛,java安装教程大多18年之前的,和现在的有一些不同,导致我按照网上的教程折腾了好几天都没安装成功,今天突然成功了,有点开心,迫不及待把自己的一些小…

蛋白质结构域的概念_Chapter1 蛋白质结构与功能2

本次回顾两大点:一、肽二、蛋白质结构————一、肽:概念:肽是由氨基酸通过肽键连接形成的线性分子。如上,该教学图涉及知识:1.肽键生成(最好能将反应式写出来)2.氨基酸残基residue3.肽的方向性N —> C4.肽命名5.肽…

css中的em 简单教程 -- 转

先附上原作的地址: https://www.w3cplus.com/css/px-to-em 习惯性的复制一遍~~~~ -------------------------------我是分界线----------------------------------- 什么是弹性布局? 用户的文字大小与弹性布局 用户的浏览器默认渲染的文字大小是“16px”&#xff0c…

ASP.NET MVC下的四种验证编程方式[续篇]

ASP.NET MVC下的四种验证编程方式[续篇] 原文:ASP.NET MVC下的四种验证编程方式[续篇]在《ASP.NET MVC下的四种验证编程方式》一文中我们介绍了ASP.NET MVC支持的四种服务端验证的编程方式(“手工验证”、“标注ValidationAttribute特性”、“让数据类型实现IValida…

axios 配置loading_用Axios Element 实现全局的请求 loading

Kapture 2018-06-07 at 14.57.40.gif背景业务需求是这样子的,每当发请求到后端时就触发一个全屏的 loading,多个请求合并为一次 loading。现在项目中用的是 vue 、axios、element等,所以文章主要是讲如果使用 axios 和 element 实现这个功能。…

请求的内容似乎是脚本,因而将无法有静态文件处理程序来处理(http error 404.17、0x80070032、IIS7)...

由于工作需要,需要在在一台新的服务器上迁移 .Net Framwork 4.0的Web项目 环境:Windows Server 2008 SP2,IIS 7,.Net FrameWork 4.0,Classic 由于项目正常在老服务器上运行,所有直接将项目拷贝过来部署在IIS上面,配置后链接字符串等等之后访问…

ZooKeeper管理分布式环境中的数据

Reference: http://www.cnblogs.com/wuxl360/p/5817549.html 本节本来是要介绍ZooKeeper的实现原理,但是ZooKeeper的原理比较复杂,它涉及到了paxos算法、Zab协议、通信协议等相关知 识,理解起来比较抽象所以还需要借助一些应用场景&#xff0…

企业级应用架构(三)三层架构之数据访问层的改进以及测试DOM的发布

在上一篇我们在宏观概要上对DAL层进行了封装与抽象。我们的目的主要有两个:第一,解除BLL层对DAL层的依赖,这一点我们通过定义接口做到了;第二,使我们的DAL层能够支持一切数据访问技术,如Ado.net,EF,linq To Sql&#x…

ctr多少正常_亚马逊广告ctr多少才算合格,如何提升亚马逊CTR

亚马逊广告ctr多少才算合格,如何提升亚马逊CTR很多亚马逊卖家并不重视点击率,其实点击率和转化率是同样重要的。好的点击率可以提升产品listing的流量,间接影响转化,促进销量。一般来说亚马逊点击率多少算正常呢?CTR全…

在linux中,我为什么不能安装VMware Tools?

在linux中,我为什么不能安装VMware Tools? 应该是操作不正确导致,以下为linux安装VMware Tools的方法。 1、在安装Linux的虚拟机中,单击“虚拟机”菜单下的“安装Vmware-Tools” 2、出现以下菜单,证明Vmware-Tools的光…

判断 CGRect是否“为空”

2019独角兽企业重金招聘Python工程师标准>>> property (nonatomic, assign) CGRect prototypeRect; -----这样的声明应该没有问题的,的if(!self.prototypeRect)报错是因为 CGRect是结构体,不能作非nil判断,你可以利用self.protot…

Javascript学习总结 - JS基础系列 二

简述 本系列将持续更新Javascript基础部分的知识,谁都想掌握高端大气的技术,但是我觉得没有一个扎实的基础,我认为一切高阶技术对我来讲都是过眼云烟,要成为一名及格的前端工程师,必须把基础打扎实了。我也想展翅高飞&…

mysql 可逆编码_使用MD5编码实现数据库用户密码字段的加密

1 前言 众所周知,MD5是目前应用最多的密码保护方法,该编码传说为不可逆加密编码<也就是说,永运无法倒算原码>。 使用MD5加密用户的操作密码,可以有效防止系统维护人员直接进入数据库时出现系统安全漏洞&…

Unity3D 与 objective-c 之间数据交互。iOS SDK接口封装Unity3D接口

原地址:http://www.cnblogs.com/qingjoin/p/3638915.html Unity 3D 简单工程的创建。与Xcode 导出到iOS 平台请看这 Unity3D 学习 创建简单的按钮、相应事件 Unity C# 代码 using UnityEngine; using System.Collections; using System.Runtime.InteropServices;pu…

开源代码ViewPageIndicator的使用

1. 导入Android studio 使用SlidingMenu的方式导入Android studio不行&#xff0c;不知道为何&#xff0c;过会懂了再写上 2. 代码 activity_main.xml <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.…

[unity3d]手游资源热更新策略探讨

原地址&#xff1a;http://blog.csdn.net/dingxiaowei2013/article/details/20079683 我们学习了如何将资源进行打包。这次就可以用上场了&#xff0c;我们来探讨一下手游资源的增量更新策略。注意哦&#xff0c;只是资源哦。关于代码的更新&#xff0c;我们稍后再来研究。理论…

PostgreSQL学习手册(二) 模式(Schema)

2019独角兽企业重金招聘Python工程师标准>>> 一个数据库包含一个或多个命名的模式&#xff0c;模式又包含表。模式还包含其它命名的对象&#xff0c;包括数据类型、函数&#xff0c;以及操作符。同一个对象名可以在不同的模式里使用而不会导致冲突&#xff1b; 比如…

软件工作第4次作业

软件工作第4次作业 信管141 宋乃佳 1425052010 基于我们列出的 7 条UX评价准则&#xff0c;分析“师路南通” 在用户体验设计方面让你觉得满意的地方&#xff08;不少于2点&#xff09;&#xff1b;&#xff08;20分&#xff09;&#xff0c;请陈述理由。 同样&#xff0c;分析…