当前位置: 首页 > news >正文

java类加载时机与过程

开篇

最近刚刚看了Uber开源的JVM Profiler的源码,对里面的修改字节码的流程有了一定的认识,刚好之前看到网上有人写了一篇关于java类加载时机与过程的文章,想了想决定把两者合并起来写一下。概念比较基础,有兴趣的可以看看。


类加载过程

img_f8a6b3625de99b06dc53b8b3a694b4b3.png
  • 加载
 加载(Loading)阶段是“类加载”(Class Loading)过程的第一个阶段,在此阶段,虚拟机需要完成以下三件事情:

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

加载阶段即可以使用系统提供的类加载器在完成,也可以由用户自定义的类加载器来完成。
加载阶段与连接阶段的部分内容(如一部分字节码文件格式验证动作)是交叉进行的,加载阶段尚未完成,连接阶段可能已经开始。
  • 验证
验证是连接阶段的第一步,这一阶段的目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。
  • 准备
准备阶段是为类的静态变量分配内存并将其初始化为默认值,这些内存都将在方法区中进行分配。
准备阶段不分配类中的实例变量的内存,实例变量将会在对象实例化时随着对象一起分配在Java堆中。

public static int value=123;
在准备阶段value初始值为0 。在初始化阶段才会变为123 。
  • 解析
解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。
  • 初始化
类初始化是类加载过程的最后一步,前面的类加载过程,除了在加载阶段用户应用程序可以通过自定义类加载器参与之外,其余动作完全由虚拟机主导和控制。
到了初始化阶段,才真正开始执行类中定义的Java程序代码。

初始化阶段是执行类构造器<clinit>()方法的过程。
<clinit>()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块(static{}块)中的语句合并产生的。

类初始化时机:
  1、创建类的实例
  2、访问类的静态变量(除常量【被final修辞的静态变量】原因:常量一种特殊的变量,因为编译器把他们当作值(value)而不是域(field)来对待。
  3、访问类的静态方法
  4、反射如(Class.forName("my.xyz.Test"))
  5、当初始化一个类时,发现其父类还未初始化,则先出发父类的初始化
  6、虚拟机启动时,定义了main()方法的那个类先初始化


一个案例

class SingleTon {
    private static SingleTon singleTon = new SingleTon();
    public static int count1;
    public static int count2 = 0;
 
    private SingleTon() {
        count1++;
        count2++;
    }
 
    public static SingleTon getInstance() {
        return singleTon;
    }
}
 
public class Test {
    public static void main(String[] args) {
        SingleTon singleTon = SingleTon.getInstance();
        System.out.println("count1=" + singleTon.count1);
        System.out.println("count2=" + singleTon.count2);
    }
}

准备阶段

 类加载的时候在准备过程中为类的静态变量分配内存并初始化默认值 
 静态变量初始化结果
  singleton=null 
  count1=0,
  count2=0

初始化阶段

在访问类的静态方法的时候会初始化(上例子中执行SingleTon singleTon = SingleTon.getInstance())
执行赋值SingleTon singleTon = new SingleTon()
private SingleTon() {
        count1++;  //count1在加载过程中初始化为0,count1++变为1
        count2++;  //count2在加载过程中初始化为0,count2++变为1
}

执行赋值count2=0 后 count2变为0

执行结果
count1=1
count2=0

java agent拦截阶段

可以在加载java文件之前做拦截把字节码做修改


参考文章

从一道面试题来认识java类加载时机与过程
java agent基础原理

相关文章:

  • 设计模式走一遍---观察者模式
  • 我发起了一个 .Net 平台上的 产生式编程 开源项目 GP.Net
  • windows远程连接报:身份错误,函数不支持的解决办法
  • Docker 笔记(2):Dockerfile
  • promise原理就是这么简单
  • EXE文件执行过程中发生了什么?
  • MathExam小学一二年级计算题生成器V1.0
  • 建设银行无人银行开业,铁饭碗是属于程序员的
  • Java 集合系列-第八篇-Map架构
  • Redhat7.0下部署NFS服务器
  • 网络,NFS
  • 服务器目录权限
  • LAMP搭建
  • 自动生成指定特征的数独题目(未完待续)
  • 学习python必备的学习网站
  • DOM的那些事
  • ES6核心特性
  • Java 11 发布计划来了,已确定 3个 新特性!!
  • Linux中的硬链接与软链接
  • macOS 中 shell 创建文件夹及文件并 VS Code 打开
  • Netty+SpringBoot+FastDFS+Html5实现聊天App(六)
  • SpiderData 2019年2月16日 DApp数据排行榜
  • underscore源码剖析之整体架构
  • zookeeper系列(七)实战分布式命名服务
  • 阿里云容器服务区块链解决方案全新升级 支持Hyperledger Fabric v1.1
  • 不发不行!Netty集成文字图片聊天室外加TCP/IP软硬件通信
  • 前端面试之CSS3新特性
  • 实现简单的正则表达式引擎
  • 数据科学 第 3 章 11 字符串处理
  • 文本多行溢出显示...之最后一行不到行尾的解决
  • 小而合理的前端理论:rscss和rsjs
  • 用Python写一份独特的元宵节祝福
  • 格斗健身潮牌24KiCK获近千万Pre-A轮融资,用户留存高达9个月 ...
  • #android不同版本废弃api,新api。
  • (2)Java 简介
  • (Redis使用系列) Springboot 使用Redis+Session实现Session共享 ,简单的单点登录 五
  • (vue)el-checkbox 实现展示区分 label 和 value(展示值与选中获取值需不同)
  • (备忘)Java Map 遍历
  • (二)换源+apt-get基础配置+搜狗拼音
  • (附源码)SSM环卫人员管理平台 计算机毕设36412
  • (附源码)ssm考试题库管理系统 毕业设计 069043
  • (原創) 系統分析和系統設計有什麼差別? (OO)
  • (转)socket Aio demo
  • (转)一些感悟
  • (转)总结使用Unity 3D优化游戏运行性能的经验
  • 、写入Shellcode到注册表上线
  • ./和../以及/和~之间的区别
  • .NET/C# 避免调试器不小心提前计算本应延迟计算的值
  • .NET/C# 阻止屏幕关闭,阻止系统进入睡眠状态
  • .netcore 如何获取系统中所有session_如何把百度推广中获取的线索(基木鱼,电话,百度商桥等)同步到企业微信或者企业CRM等企业营销系统中...
  • .net流程开发平台的一些难点(1)
  • [].shift.call( arguments ) 和 [].slice.call( arguments )
  • [AIGC] Nacos:一个简单 yet powerful 的配置中心和服务注册中心
  • [CF482B]Interesting Array
  • [DP 训练] Longest Run on a Snowboard, UVa 10285