一、类的生命周期
1.加载
①找到需要加载的类并把类的信息加载到jvm的方法中,然后在堆区中实例化一个java.lang.Class对象,作为方法区中这个类的信息入口
②类的加载方式
据类的全路径名找到相应的class文件,然后从class文件中读取文件内容
从jar文件中读取
从网络中获取
根据一定的规则实时生成,比如设计模式中的动态代理,就是根据相应的类自动生成它的代理类
从非class文件中获取,其实这与直接从class文件中获取的方式本质相同
2.连接
①验证:进行对类的合法性效验。会对比字节码格式,变量与方法的合法性,数据类型的有效性,继承与实现的规范性等等进行检查,确保加载的类能够正常的被jvm所正常运行
②准备:==为类的静态变量分配内存,并设为jvm默认的初始值;==对于非静态的变量,则不会为它们分配内存。简单说就是分内存,赋初值。
③解析:这一阶段的任务就是把常量池中的符号引用转换成直接引用
3.初始化
①类的初始化阶段是类加载的过程的最后一步。而也是到了该阶段,才真正开始执行类中定义的java程序代码(字节码),之前的动作都由虚拟机主导
②jvm对类的加载时机没有明确的规范,但对类的初始化时机有:只有当类被直接引用的时候,才会触发类的初始化。类被直接引用的情况有以下几种
通过以下几种方式
new 关键字创建对象
读取或设置类的静态变量
调用类的静态方法
通过反射执行
初始化子类的时候,会触发父类的初始化
作为程序入口直接运行时(main())
接口实现类的初始化的时候,会触发直接或间接实现的所有接口初始化
4.类的使用
①当引用了一个类的静态变量,而该静态变量继承自父类,不引起初始化
②定义一个类的数组,不会引起该类的初始化
③当引用一个类的常量时,不会引起该类的初始化
5.卸载
①该类所有实例都已经被回收,也就是java堆中不存在该类的任何实例
②加载该类的ClassLoader已经被回收
③该类对于Java.lang.Class对象没有任何地方被引用,无法在任何地方通过反射访问该类的方法
二、java子类实例化对象的过程
1.子类实例化是否会实例化父类?
不会,父类在子类实例化过程中并没有被实例化,java中new子类没有实例化父类,只是调用父类的构造方法初始化了,子类从父类继承来的属性,这个调用时子类的对象调用父类的构造方法,而子类自己的构造方法完成对自己属性初始化(这里的初始化是指我们在内存分配完了,虚拟机初始化之后,我们按自己的要求进行初始化)
2.子类对象实例化全过程
1.当最底层子类实例化对象时,它的父类,父类的父类……到Object类的所有类的构造器都会被调用,只不过当一个类拥有多个构造器时,调用的是其中一个
2.子类构造器内,默认调用父类的构造器:super();当有this关键字时,就不调用父类构造器了,就会调用同一个类内的其他构造器,所以一个类当有n个构造器时,仅最多有n-1个构造器内使用this关键字,最少有一个构造器调用上层父类的构造器
3.当父类重载一个构造器,则默认的无形参构造就会消失,父类又不重载另一个无形参的构造器,那么子类构造器不适用this或super关键字机会出错,因为子类构造器不使用this和super关键字,默认调用父类无形参构造器,而这构造器不存在,则会编译出错
4.当类实例化对象时,Object类的无形参构造器一定会被调用