this(new一个对象的流程) super(程序初始化顺序)

本文深入探讨了Java中构造器(this)和继承(super)的使用细节,包括构造器调用流程、对象创建过程及初始化顺序,揭示了this和super在代码中的关键作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、this的使用:

构造方法(*构造函数)就是创建对象(new可以通过调用构造函数来创建对象)。

实际的创建对象的流程是:

0)加载类(String.class)文件进入内存中

1)为对象分配空间(静态区(栈)和堆,String str=new String("hello");

检查静态区(常量区)有没有“hello”,有则直接在堆中创建对象,没有则在堆中和静态区中创建对象。

),并将对象初始值设置为0或者null。

2)为对象初始化属性值(默认、显示初始成员变量(*成员变量、内部类))

3)调用构造函数(*构造函数)

4)返回对象地址

<this>

构造器的调用在创建对象之后,因此,this代表当前对象的地址,可以通过this调用一个其他构造函数(必须放在第一行,否则编译报错)。

2、super的使用

<super>

super实现对父类的引用,如果构造函数没有显示的调用父类的构造方法(子类中有父类构造函数名),会默认创建一个super()调用父类构造器,如果父类没有默认无参数构造函数,会报错。并且super()位于构造函数的第一行。

创建一个子类对象时,先调用子类构造函数,再调用父类构造函数。函数调用使用“栈”,先将子类的构造函数放在栈底,再放入父类构造函数,执行时从栈顶取函数。(先执行父类,再执行子类)

 

则,Java程序初始化(放入栈空间之后的执行过程)的过程为:

1)父类(静态变量、静态语句块)

2)子类(静态变量、静态语句块)

3)父类(实例变量、普通语句块、构造函数)

4)子类(实例变量、普通语句块、构造函数)

整体方向:静态->(实例->构造函数)

### Java 类加载和初始化顺序 #### 一、类的加载与ClassLoader的理解 类的加载是指将类的 `.class` 文件中的二进制数据读入到内存中,并将其放在运行时的方法区之中,然后在堆区创建一个 `java.lang.Class` 对象,用来封装这些数据。这一过程由类加载器完成[^1]。 #### 二、什么时候会发生类初始化程序首次使用特定类或接口时,会触发该类或接口的初始化操作。具体来说: - **类的主动引用**:一定会发生类的初始化的情况包括通过 `new` 关键字实例化对象;访问静态变量或调用静态方法;反射方式获取类的信息等。 - **类的被动调用**:不会引起类的初始化的情形有定义子类时不自动触发父类的初始化;通过数组定义类引用也不会触发此类的初始化;引用常量表达式的值(如基本类型的常量或字符串常量池中的字符串)也不引发类的初始化。 #### 三、类加载器的作用 类加载器负责查找并装载所需的类文件至 JVM 中。它遵循双亲委派模型原则工作,即先尝试让上级加载器去处理请求,在找不到对应资源的情况下才会自己继续寻找相应的类文件[^3]。 #### 四、动态创建对象执行方法 对于新创建的对象而言,除了要经历上述提到的一系列加载流程外,还需要按照一定的顺序来进行成员变量赋初值以及构造函数内的逻辑运算。这涉及到字段声明处赋予默认值/显式指定初始值的动作,还有可能存在的静态代码块与非静态代码块被执行的过程[^2]。 #### 五、成员初始化 成员变量可以在声明的同时给予初始值,也可以不给定任何初始值而采用系统的默认值。如果存在多个同名但不同作用域下的局部变量,则最内层的作用范围优先级最高[^4]。 #### 六、构造器初始化 每当创建一个新的对象实例时都会调用对应的构造器来设置一些必要的状态信息。需要注意的是,即使是没有显示写出无参构造器,默认情况下也会有一个隐含的存在着。 #### 七、初始化顺序 考虑下面的例子可以更好地理解整个初始化序列是如何工作的: ```java // 定义两个类 A 和 B,其中 B 继承自 A class A { public static String sStaticField = "A's Static Field"; static { System.out.println("Initializing Class A"); } } class B extends A { public static String sSubClassField = "B's Sub-Class Field"; static { System.out.println("Initializing Class B"); } private int instanceVar; { this.instanceVar = 9; System.out.println("Instance Block of B Executed."); } public B(){ super(); System.out.println("Constructor of B Called."); } } ``` 在这个例子中,假设我们只执行了如下语句: ```java public class Main{ public static void main(String[] args){ new B(); // 创建 B 的实例 } } ``` 那么输出将会是这样的: ``` Initializing Class A Initializing Class B Instance Block of B Executed. Constructor of B Called. ``` 这是因为首先会加载基类 `A` 并打印出 `"Initializing Class A"` ,接着再加载派生类 `B` 打印出 `"Initializing Class B"` 。之后是非静态代码块里的内容被执行,最后才是构造器被调用[^5]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值