郑州大学网页设计与网站建设,金华网站建设luopan,手机网站建设找哪家好,音乐网站要怎么做3.编译期处理
编译期优化称为语法糖 3.1 默认构造器 3.2 自动拆装箱
java基本类型和包装类型之间的自动转换。 3.3泛型集合取值 在字节码中可以看见#xff0c;泛型擦除就是字节码中的执行代码不区分是String还是Integer了#xff0c;统一用Object. 对于取出的Object…3.编译期处理
编译期优化称为语法糖 3.1 默认构造器 3.2 自动拆装箱
java基本类型和包装类型之间的自动转换。 3.3泛型集合取值 在字节码中可以看见泛型擦除就是字节码中的执行代码不区分是String还是Integer了统一用Object. 对于取出的Object这个checkcast用于转换Object为Integer类型。 可以看见局部变量类型表里面包含了方法参数的泛型信息。 泛型反射 输出里拿到了两个参数的原始类型List和Map。
这里除了能拿到方法参数上的泛型信息还能拿到返回值的泛型信息。 3.4可变参数 3.5 foreach 循环 如果遍历的是List集合生成代码如下 3.6 switch 字符串 一个switch变成了两个switch通过字符串的哈希码进行匹配。 3.7 switch 枚举 3.8枚举类 这玩意的本质也是一个class,里面的两个值就是这个class的两个实例对象。
跟普通类的最大区别普通类的实例个数是无穷枚举类的实例个数有限。
转换后因为这个class不能再被继承所以加上了final关键字。并继承了一个支持泛型的枚举父类 3.9 try-with-resources
主要作用是简化资源的关闭只要按照下面的语法在try后面括号内进行资源的创建就可以省略finally中的资源关闭。 编译器会帮助生成finally的代码。 3.10方法重写时的桥接方法 编译器多加了一个真正的方法重写在其内部调用了我们重写的m方法返回Integer。
该合成方法可以重名只在jvm内部使用。 3.11匿名内部类 额外生成的这个类是外部类名称$1 因为转换后的类已经变成有两个方法了所以要加多一个参数接收传进来的x,并传给run方法。
匿名内部类引用外部局部变量必须是finall的原因。 4.类加载阶段
类的生命周期有如下阶段 加载Loading在加载阶段类的字节码被加载到JVM的内存中。这包括从文件系统、网络等位置加载类的字节码并将其转换为JVM能够理解的数据结构。加载阶段的结果是在方法区Method Area创建一个代表该类的Class对象。 连接Linking连接阶段将加载的类与其他类和资源进行关联以便正确地解析和执行。连接阶段可以进一步分为三个子阶段 验证Verification验证阶段确保类的字节码符合JVM规范并且不会引发安全问题或错误。准备Preparation在准备阶段为类的静态变量分配内存并初始化为默认值例如数值类型初始化为0引用类型初始化为null。解析Resolution解析阶段将符号引用如类、方法、字段的符号名称解析为直接引用以便在后续的执行阶段中能够准确访问相关内容。 初始化Initialization初始化阶段是类生命周期中的关键阶段它负责执行类的静态初始化代码块和静态变量的赋值。在该阶段JVM确保静态变量按照预期值进行初始化静态初始化块被执行构造器也可以被调用但仅限于初始化静态字段。初始化阶段标志着类已准备好被使用。 使用Usage在初始化阶段之后类就可以被使用了。这包括通过创建对象、调用类的方法、访问静态变量等方式来使用类。 卸载Unloading如果某个类不再被引用JVM可能会在某些情况下将其卸载释放对应的内存和资源。类的卸载是一个可选的过程通常由垃圾回收器决定。
需要注意的是类的生命周期可能因为具体的JVM实现、加载方式、类加载器等因素而有所不同。上述阶段的顺序和细节可能会因JVM的版本和配置而有所变化。
从大方面来说分为加载连接和初始化阶段。
4.1加载
加载阶段Java虚拟机需要完成以下三件事
通过一个类的全限定名来获取定义此类的二进制字节流。将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构。在内存中生成一个代表这个类的java.lang.Class对象作为方法区这个类的各种数据的访问入口 java_mirror属性起到桥梁作用c数据结构和java的桥梁java的对象要想访问它的Klass信息不能直接访问要通过java_mirror镜像访问。
常用的XXX.class不是指Klass,而是java_mirror。
如果该类有父类或接口没有加载会先加载父类和接口才会轮到这个类加载。 类的字节码被加载到元空间构成instanceKlass数据结构。其中 _java_mirror是java的类镜像它持有该类在堆内存中的地址堆内存中的类对象也持有instanceKlass在本地内存中的地址。
用new关键字创建person的实例对象时每个实例对象都有自己的对象头是16个字节其中八个字节对应着对象的class地址。
如果要通过实例对象获取其class信息会先访问对象头然后通过class地址先找到堆中的类对象再间接找到元空间的instanceKlass.
扩展我们通过类模版创建对象后此时对象头中有8个字节对应 类模版class在堆内存的地址如果想调用普通方法或者 getset 方法就要通过对象头中类模版地址找到元空间中的instanceKlass再通过instanceKlass获取 fieldmethods 信息。 instanceKlass元空间中的类元信息 instanceKlass 是在HotSpot虚拟机的元空间中存储的一种数据结构它表示已加载的类的元信息。这个数据结构包含了类的名称、修饰符、方法、字段、父类、接口等信息。在HotSpot的内部实现中instanceKlass 是用于支持虚拟机的运行时类型信息和方法的查找调用等操作。它和Java中的Class对象类似但是在内存结构上可能有所不同。 Class对象堆中的类元信息 在Java中每个已加载的类都有一个与之对应的Class对象用来表示类的元信息。Class对象存储了类的名称、修饰符、方法、字段、父类、接口等信息也支持反射操作。这个Class对象是存在于堆内存中的它是Java反射机制的核心。
虽然instanceKlass 和Class对象在某种程度上都用来表示类的元信息但它们的实现和作用在内部有所区别。instanceKlass 主要在HotSpot虚拟机的内部实现中使用而Class对象是Java编程中常用的反射机制的一部分。这两者的关系是instanceKlass 是HotSpot虚拟机内部用于支持类的元数据管理的一部分而Class对象则是供Java程序员在运行时获取和操作类信息的关键工具。
4.2 链接
验证Verification
验证阶段确保类的字节码符合JVM规范并且不会引发安全问题或错误。
随便修改一下编译后class文件就会验证报错。
准备Preparation
在准备阶段为类的静态变量分配内存并初始化为默认值例如数值类型初始化为0引用类型初始化为null。
静态变量跟类对象存储在一起都在堆中。早期的jvm静态变量跟着instanceKlass存储在方法区。
public class Example {static int staticVariable 42;public static void main(String[] args) {Example instance new Example();// 通过类名访问静态变量System.out.println(Example.staticVariable); // 输出: 42// 通过类对象访问静态变量System.out.println(instance.staticVariable); // 输出: 42}
}在这个示例中通过类对象 instance 访问了静态变量 staticVariable并且与通过类名直接访问的结果相同。
需要注意的是尽管在JDK 8及之后版本中支持通过类对象访问静态变量但这并不是很常见的用法通常还是使用类名来访问静态变量更加清晰和符合约定。
通过上面代码可以知道a只是分配了空间没有赋值。b在准备阶段没有赋值只是准备好了语句还要等到初始化阶段而cd已经赋完值了。
解析Resolution
解析阶段将常量池中的符号引用如类、方法、字段的符号名称解析为直接引用以便在后续的执行阶段中能够准确访问相关内容。
默认情况下类的加载都是懒惰式的用到了类C,没有用到类D的话类D不会主动加载。 上面代码准备了一个类加载器调用了loadClass方法加载类C但这只会,进行类C的加载不会进行解析和初始化。
new C()会进行C的加载解析和初始化并把D也给加载解析和初始化。
类D未解析的情况时
在HSDB中连接到该进程上只能看见有类C的存在类C的常量池中可以看见类D前面标注了常量_未经解析类仅仅是个符号。
new C(),类D被解析时
现在C,D都有了并且类C的常量池里面也知道了类D的内存地址。
这就是一个符号引用一个直接引用。解析阶段就是让符号引用变成直接引用。
4.3 初始化
初始化阶段是类生命周期中的关键阶段它负责执行类的静态初始化代码块和静态变量的赋值。在该阶段JVM确保静态变量按照预期值进行初始化静态初始化块被执行构造器也可以被调用但仅限于初始化静态字段。初始化阶段标志着类已准备好被使用。
cinit()v方法
初始化即调用cinit()V虚拟机会保证这个类的[构造方法]的线程安全
发生的时机
加载阶段就已经生成了mirror_class也就是类.class所以不会触发静态代码块。数组存的也还是Minor对象在加载时就创建了。
4.4 练习 a,b都不会导致类的初始化两个都是final的基本类型在准备阶段就已经赋值了。
c会导致类的初始化这是包装类型底层会调用Integer.valueOf(20).
字节码的静态代码块中能看见有关C的操作说明C是在初始化阶段进行。 单例模式就是保证在jvm中这个类的对象只有一个。
静态内部类的好处是可以访问外部类的资源构造方法。所以私有构造方法也可以调用。
只要不调getIntance()就不会触发LazyHolder的加载连接和初始化。