Java字节代码:byte[]Java类在JVM的表示形式:Class类的对象;
Java源代码被编译成class字节码 :
了解java执行顺序,为代码书写提供最佳的执行顺序,优化代码-1.jpg (11.41 KB, 下载次数: 0)
2018-10-13 11:03 上传
Java字节代码 --> Class类的对象:
加载:把Java字节码byte[]转换成JVM中的java.lang.Class类的对象;链接:Java类的链接指的是将Java类的二进制代码合并到JVM的运行状态之中的过程。初始化:主要是执行静态代码块和初始化静态域;
Java类的加载
作用
把Java字节码转换成JVM中的java.lang.Class类的对象;
通过一个类的全限命名获取描述此类的二进制字节流;将这个字节流所代表的静态存储结构保存为体例区的运行时数据结构;在java堆中生成一个代表这个类的java.lang.Class对象,作为拜候体例区的入口;
类加载器分类
启动类加载器(Bootstrap ClassLoader):负责加载 JAVA_HOMElib 目录中的,或通过-Xbootclasspath参数指定路径中的,且被虚拟机认可(按文件名识别,如rt.jar)的类;扩展类加载器(Extension ClassLoader):负责加载 JAVA_HOMElibext 目录中的,或通过java.ext.dirs系统变量指定路径中的类库;应用程序类加载器(Application ClassLoader):负责加载用户路径(classpath)上的类库;
了解java执行顺序,为代码书写提供最佳的执行顺序,优化代码-2.jpg (16.67 KB, 下载次数: 0)
2018-10-13 11:03 上传
双亲委派模型工作过程:
当一个类加载器收到类加载任务,优先交给其父类加载器去完成,因此最终加载任务城市传递到顶层的启动类加载器,只有当父类加载器无法完成加载任务时,才会测验考试执行加载任务。
双亲委派模型有什么好处?
好比位于rt.jar包中的类java.lang.Object,无论哪个加载器加载这个类,最终都是委托给顶层的启动类加载器进行加载,确保了Object类在各种加载器环境中都是同一个类。
重要特征
条理组织结构:每个类加载器都有一个父类加载器,形成tree结构;代办署理模式:一个类加载器既可以自己完成Java类的界说工作,也可以代办署理给其它的类加载器来完成;
Java类的链接
Java类的链接指的是将Java类的二进制代码合并到JVM的运行状态之中的过程。
包含的步调:
验证:确保Java类的二进制暗示在结构上是完全正确的,主要包含格式验证、元数据验证、字节码验证和符号引用验证;准备:建立Java类中的静态域,并将这些域的值设为默认值;在准备阶段,为类变量(static修饰)在体例区中分派内存并设置初始值。private static int var = 100;准备阶段完成后,var值为0,而不是100。在初始化阶段,才会把100赋值给val,可是有个特殊情况:private static final int VAL= 100;在编译阶段会为VAL生成ConstantValue属性,在准备阶段虚拟机会根据ConstantValue属性将VAL赋值为100。解析:解析阶段是将常量池中的符号引用替换为直接引用的过程,解析过程可能致使其他的Java类被加载;
符号引用使用一组符号来描述所引用的目标,可以是任何形式的字面常量,界说在Class文件格式中。直接引用可以是直接指向目标的指针、相对偏移量或则能间接定位到目标的句柄。
Java类的初始化
初始化阶段是执行类构造器clinit体例的过程,clinit体例由类变量的赋值动作和静态语句块依照在源文件呈现的顺序合并而成,该合并操作由编译器完成。
体例clinit对类或接口不是必须的,如果一个类中没有静态代码块,也没有静态变量的赋值操作,那么编译器不会生成体例clinit体例与实例构造器不合,不需要显式的挪用父类的为了避免屡次执行clinit,虚拟机会确保clinit体例在多线程环境下被正确的加锁同步执行,如果有多个线程同时初始化一个类,那么只有一个线程能够执行clinit体例,其它线程进行阻塞期待,直到clinit执行完成。注意:执行接口的clinit体例不需要先执行父接口的clinit,只有使用父接口中界说的变量时,才会执行。
初始化过程的主要操作是执行静态代码块和初始化静态域。在一个类被初始化之前,它的直接父类也需要被初始化。
类初始化场景
虚拟机中严格规定了有且只有5种情况必须对类进行初始化。
执行new、getstatic、putstatic和invokestatic指令;使用reflect对类进行反射挪用;初始化一个类的时候,父类还没有初始化,会事先初始化父类;启动虚拟机时,需要初始化包含main体例的类;在JDK1.7中,如果java.lang.invoke.MethodHandler实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的体例句柄,并且这个体例句柄对应的类没有进行初始化;
不会触发类初始化的情况
通过子类引用父类的静态字段,只会触发父类的初始化,而不会触发子类的初始化。
了解java执行顺序,为代码书写提供最佳的执行顺序,优化代码-3.jpg (25.41 KB, 下载次数: 0)
2018-10-13 11:03 上传
输出结果为:parent init! 不会初始化Child类。
界说对象数组,不会触发该类的初始化。
了解java执行顺序,为代码书写提供最佳的执行顺序,优化代码-4.jpg (14.11 KB, 下载次数: 0)
2018-10-13 11:03 上传
常量在编译期间会存入挪用类的常量池中,素质上并没有直接引用界说常量的类,不会触发界说常量所在的类。
了解java执行顺序,为代码书写提供最佳的执行顺序,优化代码-5.jpg (20.66 KB, 下载次数: 0)
2018-10-13 11:03 上传
在编译阶段,Const类中常量A的值100存储到Init类的常量池中,这两个类在编译成class文件之后就没有联系了。
通过类名获取Class对象,不会触发类的初始化。
了解java执行顺序,为代码书写提供最佳的执行顺序,优化代码-6.jpg (27.12 KB, 下载次数: 0)
2018-10-13 11:03 上传
通过Class.forName加载指定类时,如果指定参数initialize为false时,也不会触发类初始化,其拭魅这个参数是告诉虚拟机,是否要对类进行初始化。通过ClassLoader默认的loadClass体例,也不会触发初始化动作;
更多内容回复查看:
游客,如果您要查看本帖隐藏内容请回复