烧点脑子使劲看--类的加载过程

本文涉及的产品
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: . 概述当程序需要使用到某个类时,如果该类还没有被加载到内存中,就需要先将类加载到内存中。类的加载过程分为五个阶段:加载、验证、准备、解析、初始化。

一、类的加载过程


1. 概述

当程序需要使用到某个类时,如果该类还没有被加载到内存中,就需要先将类加载到内存中。类的加载过程分为五个阶段:加载、验证、准备、解析、初始化。


2. 加载

“加载”是整个“类加载”过程中的一个阶段,在加载阶段Java虚拟机需要完成以下三件事:

  • 通过一个类的全限定名来获取定义此类的二进制字节流。
  • 将这个字节流所代表的的静态存储结构转化为方法区的运行时数据结构。
  • 在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据访问入口。

在这三个动作中,获取类的二进制字节流,对于开发人员来说,是可控性最强的阶段,不仅可以获取本地class文件的二进制流,还可以从ZIP包中读取、从网络中获取、在运行时动态生成(如动态代理技术)等。

加载阶段可以使用Java虚拟机内置的启动类加载器来完成,也可以使用自定义的类加载器,来自定义获取类的二进制字节流的方式。


3. 验证

验证是连接的第一步,目的是确保Class文件中的字节流符合《Java虚拟机规范》的要求,保证这些信息作为代码运行后不会危害虚拟机自身的安全。验证阶段主要分成四个部分:

  • 文件格式验证

在文件格式验证阶段,Java虚拟机需要验证Class文件的格式是否符合规范并且能够被当前的虚拟机版本处理,主要目的是为了确保输入的字节流能够被正确解析并储存于方法区。在通过这个阶段的验证后,字节流就会进入方法区中进行储存,后续的三个验证阶段都是基于方法区的存储结构上进行的,不会再读取、操作字节流了。

  • 元数据验证

这个阶段主要是对类的元数据信息进行语义校验,如这个类是否有父类(除了Object类,其它所有类都应该有父类)、这个类是否继承了不允许继承的类(被final修饰的类)、是否实现了接口中的方法等等。

  • 字节码验证

字节码验证的目的是确保程序语义是合法的、符合逻辑的,在这个阶段会对方法体进行校验分析,保证类的方法在运行时不会做出危害虚拟机安全的行为。

  • 符号引用验证

符号引用验证发生在解析阶段虚拟机将符号引用转化为直接引用的时候,主要验证该类是否缺少或引用了禁止引用的外部类、方法、字段等资源。如果无法通过符号引用验证,将抛出一个IncompatibleClassChangeError的子类异常,如:IllegalAccessErrorNoSuchFieldErrorNoSuchMethodError等异常。

验证阶段对虚拟机的类加载机制来说是很重要的,但不是必要的,因为验证结果只有通过和不通过,只要通过了对后续的代码执行都不会有影响,如果项目中的类都已经被验证通过了,可以在生产环境中考虑使用参数-Xverify:none将验证阶段关闭,加快类的加载速度。


4. 准备

准备阶段主要是为类中定义的变量(被static修饰的变量)分配内存空间并赋初值,注意这里的初值会分成两种情况:

  • 没有被final修饰的变量
public static int i = 123;

此时的i将被赋值为0,而不是123

  • 被final修饰的变量
public static final int i = 123;

这种情况的i将直接赋值为123


5. 解析

解析阶段是Java虚拟机将常量池中的符号引用替换成直接引用的过程。

  • 符号引用
    符号引用就是一个字符串,只要能够定位到目标即可, 如下面的a就是符号引用:
String a = "abc";
System.out.println(a);
  • 直接引用 直接引用可以理解为指向目标的内存地址,或者一个偏移量,比如类方法、类变量的直接引用是直接指向方法区的指针,而实例方法、实例变量的直接引用则是实例方法或实例变量相对于实例起始地址的偏移量。

在解析阶段,JVM会把所有的类名、字段名、方法名的符号引用替换成直接引用


6. 初始化

这个阶段是对类变量初始化,执行类构造器的过程,在准备阶段,static修饰的变量会被赋初值,在初始化阶段,将对这些变量的值初始化。如果一个类中存在多个静态变量和静态代码块按照源文件中出现的顺序执行。

目录
相关文章
|
8月前
|
存储 缓存 NoSQL
不扒瞎,这个程序让我从150s优化到了5s
在优化一个业务开发组的生产问题时,发现销售管理系统查询数据延迟高达2-3分钟。问题根源在于,程序在for循环中频繁读取Redis大KEY数据,导致性能下降。解决方案是采用本地缓存HutoolCache,将耗时降至毫秒级别。此外,还对RedisTemplate配置进行了研究,Jackson2JsonRedisSerializer在序列化时包括了所有字段,即使字段值为null,增加了数据体积。通过对ObjectMapper的调整,仅序列化非空字段,可以显著提升redis读取性能。本文同时还提醒我们在使用Redis时要注意大对象缓存,强调了正确使用和配置缓存以及避免大对象存储的重要性。
79 5
|
Go
腥风血雨中,这招救了我的代码!
腥风血雨中,这招救了我的代码!
70 0
|
存储 机器学习/深度学习 监控
我是傻x,被迫看了 1 天源码,千万别学我!
大家好,我是零一,之前一直很忙,业余时间的输入和输出都 24k铝合金人眼可见 得下降,这不最近上海疫情严重么,算了一下居家办公也已经将近 1个月了,这才有些许时间学习,所以最近也是一直在鼓捣点新东西,不为别的,主要是想再多输入一些新的知识
190 0
我是傻x,被迫看了 1 天源码,千万别学我!
|
消息中间件 Kubernetes Cloud Native
记一次内部分享——瞎扯淡
记一次内部分享——瞎扯淡
记一次内部分享——瞎扯淡
|
存储 算法 安全
烧点脑子使劲看--对象详细讲解
当Java虚拟机遇到一条new字节码指令时,首先会检查这个指令的参数是否能在常量池中定位到一个类的符号引用,并检查这个符号引用所代表的类是否已经被加载,如果没有,就必须先将就该类加载到内存中,具体过程见:
107 0
|
算法 Java
|
Python
又烧脑又炫技还没什么用,在代码里面打印自身
又烧脑又炫技还没什么用,在代码里面打印自身
229 0
又烧脑又炫技还没什么用,在代码里面打印自身
|
存储 缓存 NoSQL
没弄懂深浅拷贝你也敢用缓存?
而其中,最简单的肯定就是第一种服务内部的缓存了。强哥前些天就是用了Guava做了缓存,结果因为代码写的有些乱,就出了个匪夷所思的问题。搞了半天最后才发现问题所在。在这里和大家分享一哈。
没弄懂深浅拷贝你也敢用缓存?
|
Java 容器 Spring
521我发誓读完本文,再也不会担心Spring配置类问题了(下)
521我发誓读完本文,再也不会担心Spring配置类问题了(下)
521我发誓读完本文,再也不会担心Spring配置类问题了(下)
|
IDE Java 中间件
521我发誓读完本文,再也不会担心Spring配置类问题了(上)
521我发誓读完本文,再也不会担心Spring配置类问题了(上)
521我发誓读完本文,再也不会担心Spring配置类问题了(上)

热门文章

最新文章

相关实验场景

更多