众所周知,在java里是不能给构造函数写返回值的,若是在低版本的编译器定义一个构造器写上返回值可能会报错,高版本里面他就是一个普通的方法。
但是若是构造函数没有返回值,那么好比Test t = new Test()咱们new一个对象的时候是怎么赋值的呢?java
写一段代码测试一下:函数
public class Test { public Test() { } public static void main(String[] args) { Test t = new Test(); } }
反编译一下看看:测试
Code: 0: new #5 // class com/irving/utils/baidu/Test 3: dup 4: invokespecial #6 // Method "<init>":()V 7: astore_1 8: return
从反编译的结果看 4: invokespecial #7 // Method "init": ()V,调用构造函数,V表明void无返回值,那么init表明什么含义?this
我在书里找到这样一段话:编码
在 Java 虚拟机层面上,Java 语言中的构造函数是以一个名为init的特殊实例初始化方法的形式出现的,init这个方法名称是由编译器命名的,由于它并不是一个合法的 Java 方法名字,不可能经过程序编码的方式实现。实例初始化方法只能在实例的初始化期间,经过 Java 虚拟机的 invokespecial 指令来调用, 只有在实例正在构造的时候,实例初始化方法才能够被调用访问。一个类或者接口最多能够包含不超过一个类或接口的初始化方法,类或者接口就是经过这个方法完成初始化的。这个方法是一个不包含参数的静态方法,名为clinit。这个名字也是由编译器命名的,由于它并不是一个合法的 Java 方法名字,不可能经过程序编码的方式实现。类或接口的初始化方法由 Java 虚拟机自身隐式调用,没有任何虚拟机字节码指令能够调用这个方法,只有在类的初始化阶段中会被虚拟机自身调用。spa
init表明着虚拟机调用构造函数,如今状况很明显,构造函数返回类型是void,那么它到底是怎么赋值的呢?code
咱们明白一点,方法的调用过程就是栈帧入栈和出栈的过程,栈帧随着方法的调用建立,方法结束销毁。栈帧的内部包含局部变量表、操做数栈、动态连接等。对象
局部变量表表示方法调用时候的参数传递,当一个实例方法被调用的时候,第0个局部变量存储了当前实例方法所在对象的引用(this),后续的其余参数传递至1到N的连续位置。索引
操做数栈用来准备方法调用的参数和返回结果。接口
以上面测试代码的方法来看Test t = new Test() 的调用过程:
从这个过程咱们已经看出来了,整个过程最后咱们最终拿到了new以后建立的对象引用,而且保存到局部变量表中,能够供咱们继续使用。
关注公众号:java宝典
![]()