java new一个对象的过程当中发生了什么

java在new一个对象的时候,会先查看对象所属的类有没有被加载到内存,若是没有的话,就会先经过类的全限定名来加载。加载并初始化类完成后,再进行对象的建立工做。java

咱们先假设是第一次使用该类,这样的话new一个对象就能够分为两个过程:加载并初始化类建立对象。多线程

1、类加载过程(第一次使用该类)

  java是使用双亲委派模型来进行类的加载的,因此在描述类加载过程前,咱们先看一下它的工做过程:jvm

双亲委托模型的工做过程是:若是一个类加载器(ClassLoader)收到了类加载的请求,它首先不会本身去尝试加载这个类,而是把这个请求委托给父类加载器去完成,每个层次的类加载器都是如此,所以全部的加载请求最终都应该传送到顶层的启动类加载器中,只有当父类加载器反馈本身没法完成这个加载请求(它的搜索范围中没有找到所须要加载的类)时,子加载器才会尝试本身去加载。
使用双亲委托机制的好处是:可以有效确保一个类的全局惟一性,当程序中出现多个限定名相同的类时,类加载器在执行加载时,始终只会加载其中的某一个类。优化

一、加载spa

     由类加载器负责根据一个类的全限定名来读取此类的二进制字节流到JVM内部,并存储在运行时内存区的方法区,而后将其转换为一个与目标类型对应的java.lang.Class对象实例线程

二、验证

格式验证:验证是否符合class文件规范
语义验证:检查一个被标记为final的类型是否包含子类;检查一个类中的final方法是否被子类进行重写;确保父类和子类之间没有不兼容的一些方法声明(好比方法签名相同,但方法的返回值不一样)
操做验证:在操做数栈中的数据必须进行正确的操做,对常量池中的各类符号引用执行验证(一般在解析阶段执行,检查是否能够经过符号引用中描述的全限定名定位到指定类型上,以及类成员信息的访问修饰符是否容许访问等)指针

三、准备

为类中的全部静态变量分配内存空间,并为其设置一个初始值(因为尚未产生对象,实例变量不在此操做范围内)
被final修饰的static变量(常量),会直接赋值;对象

四、解析

将常量池中的符号引用转为直接引用(获得类或者字段、方法在内存中的指针或者偏移量,以便直接调用该方法),这个能够在初始化以后再执行。
解析须要静态绑定的内容。  // 全部不会被重写的方法和域都会被静态绑定继承

  以上二、三、4三个阶段又合称为连接阶段,连接阶段要作的是将加载到JVM中的二进制字节流的类数据信息合并到JVM的运行时状态中。内存

五、初始化(先父后子)

4.1 为静态变量赋值

4.2 执行static代码块

注意:static代码块只有jvm可以调用
   若是是多线程须要同时初始化一个类,仅仅只能容许其中一个线程对其执行初始化操做,其他线程必须等待,只有在活动线程执行完对类的初始化操做以后,才会通知正在等待的其余线程。

 

由于子类存在对父类的依赖,因此类的加载顺序是先加载父类后加载子类,初始化也同样。不过,父类初始化时,子类静态变量的值也有有的,是默认值。

最终,方法区会存储当前类类信息,包括类的静态变量类初始化代码定义静态变量时的赋值语句 和 静态初始化代码块)、实例变量定义实例初始化代码定义实例变量时的赋值语句实例代码块构造方法)和实例方法,还有父类的类信息引用。

 

2、建立对象

一、在堆区分配对象须要的内存

  分配的内存包括本类和父类的全部实例变量,但不包括任何静态变量

二、对全部实例变量赋默认值

  将方法区内对实例变量的定义拷贝一份到堆区,而后赋默认值

三、执行实例初始化代码

  初始化顺序是先初始化父类再初始化子类,初始化时先执行实例代码块而后是构造方法

四、若是有相似于Child c = new Child()形式的c引用的话,在栈区定义Child类型引用变量c,而后将堆区对象的地址赋值给它

 

须要注意的是,每一个子类对象持有父类对象的引用,可在内部经过super关键字来调用父类对象,但在外部不可访问

 

补充:

经过实例引用调用实例方法的时候,先从方法区中对象的实际类型信息找,找不到的话再去父类类型信息中找。

若是继承的层次比较深,要调用的方法位于比较上层的父类,则调用的效率是比较低的,由于每次调用都要通过不少次查找。这时候大多系统会采用一种称为虚方法表的方法来优化调用的效率。

所谓虚方法表,就是在类加载的时候,为每一个类建立一个表,这个表包括该类的对象全部动态绑定的方法及其地址,包括父类的方法,但一个方法只有一条记录,子类重写了父类方法后只会保留子类的。当经过对象动态绑定方法的时候,只须要查找这个表就能够了,而不须要挨个查找每一个父类。

相关文章
相关标签/搜索