jvm类加载机制

解析阶段:可以在初始化完成之后再开始,这是为了直接java的运行时绑定(动态绑定)。注:基于HotSpot.

各个阶段的工作:

一:加载

                1.通过类的全限定名来获取此类的二进制字节流

                2.将字节流代表的静态存储结构转化为方法区运行时数据结构

                3.在内存中生成一个代表这个类的class对象,作为方法区这个类的各种数据访问入口

说明:加载和连接阶段的部分内容是交叉进行的,加载阶段尚未完成连接阶段已经开始。但这两个阶段的开始时间任然保持着固定的先后顺序。


二:连接

                验证

                        目的:确保class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。

                        检验动作:

                                        1.文件格式验证;

                                        2.元数据验证;

                                        3.字节码验证;

                                        4.符号引用验证;

                            说明:验证阶段非常重要但不是必须的,它对程序运行期没有影响。

                                       可以采用-X verifynone参数来关闭大部分类验证措施,以缩短虚拟机类加载的时间。

                准备

                        作用:正式为类变量(static)分配内存并设置类变量初始值。这时候进行内存分配的仅包括类变量,而不包括实例变量。实例变量将会在对象实例化时随着对象一起分配在队中。

                        public static int value=123;

                        注:在准备阶段的过后value的初始值为0而不是123(因为尚未执行任何java方法)。若将value声明为final那么value在准备阶段的值初始化为123而不是0.

                解析

                        作用:虚拟机将常量池内的符号引用替换为直接引用的过程。

                        解析动作:主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符⑦类符号引用。


三:初始化

                 作用:初始化阶段才开始真正执行类中定义的java程序代码。

                             根据程序员通过程序制定的主管计划去初始化类变量和其他资源。(执行类构造器方法)。

静态语句块只能访问到定义在静态语句块之前的变量,定义在它之后的变量,在前面的静态语句块可以赋值,但是不能访问。

同一个类加载器下,一个类型只会初始化一次。

                虚拟机规范规定了5种情况必须对类进行初始化

                    1.遇到newgetstaticputstaticinvokestatic这四条字节码指令时必须进行初始化(如果没有初始化的话)。

                            常见的java代码常景-

                                -new关键之实例化对象的时候

                                -读取或设置一个类的静态字段时

                                -调用一个类的静态方法的时候

                    2.使用lang.reflect包的方法对类进行反射调用的时候。如果类没有进行初始化则必须进行初始化。

                    3.当初始化一个类的时候,如果发现父类还没有进行初始化,则需要先触发其父类初始化。

                    4.当虚拟机启动时,用户需要制定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。

                    5.当使用jdk1.7动态语言支持时...

  so :1通过子类引用父类的静态字段不会导致子类初始化。2.通过数组定义来引用类,不会触发此类的初始化。

         3.常量在编译阶段会存入调用类的常量池中,本质上并没有直接引用到定义常量的类,因此不会触发定义常量的类初始化。


四:使用


五:卸载



猜你喜欢

转载自blog.csdn.net/sjp_359056/article/details/80209426