JJVM类的加载过程

类的加载过程

一个java文件从被加载到被卸载这个生命过程,总共要经理五个阶段,JVM将类加载过程分为:(加链初使卸

1. 加载

首先通过一个类的全限定名来获取此类的二进制字节流;其次将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构;最后在java堆中生成一个代表这个类的Class对象,作为方法区这些数据的访问入口。总的来说就是查找并加载类的二进制数据。

2. 链接

验证:确保被加载类的正确性,字节码开头必须为cafebabe

准备:为类的静态变量分配内存,并将其初始化为默认值

解析:把类中的符号引用转换为直接引用

3. 类的初始化

1> 类什么时候才被初始化
  • 创建类的实例,也就是new一个对象

  • 访问某个类或接口的静态变量,或者对该静态变量赋值

  • 调用类的静态方法

  • 反射(Class.forName(类路径))

  • 初始化一个类的子类(会首先初始化子类的父类)

  • JVM启动时标明的启动类,即文件名和类名相同的那个类

    另外:

    1、类中的static final 修饰的基本数据类型及String类型成员变量,被访问是不会触发类初始化的;

    2、静态内部类也是被访问不会触发类初始化;

2>类的初始化顺序
  1. 如果这个类还没有被加载和链接,那先进行加载和链接

  2. 假如这个类存在直接父类,并且这个类还没有被初始化(注意:在一个类加载器中,类只能初始化一次),那就初始化直接的父类(不适用于接口)

  3. 加入类中存在初始化语句(如static变量和static块),那就一次执行这些初始化语句

  4. 总的来说,初始化顺序依次是:(静态变量、静态初始化块)->(变量、初始化块)-> 构造器

如果有父类,则顺序是:

父类static方法 -> 子类static方法 -> 父类构造方法 -> 子类构造方法

4. 类的加载

类的加载指的是将类的.class文件中的二进制数据读入内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个这个类的java.lang.Class对象。

先将类的字节码文件读入内存,放入方法区中,然后在堆中创建一个该类的Class对象

如:

image-20230812140421672

image-20230812140617240

类的加载的最终产品是位于堆中的Class对象。Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口。加载类的方式有以下几种:

  1. 从本地系统直接加载

  2. 通过网络下载.class文件

  3. 从zip,jar等归档文件中加载.class文件

  4. 从专有数据库中提取.class文件

  5. 将Java源文件动态编译为.class文件(服务器)

5.加载器

JVM的类加载是通过ClassLoader及其子类来完成的,类的层次关系和加载顺序可以由下图来描述:

image-20230812141155707

加载器介绍:

1)BootstrapClassLoader(启动类加载器)   负责加载$JAVA_HOME中jre/lib/rt.jar里所有的class,加载System.getProperty(“sun.boot.class.path”)所指定的路径或jar。 2)ExtensionClassLoader(标准扩展类加载器)   负责加载java平台中扩展功能的一些jar包,包括$JAVA_HOME中jre/lib/*.jar或-Djava.ext.dirs指定目录下的jar包。载System.getProperty(“java.ext.dirs”)所指定的路径或jar。 3)AppClassLoader(系统类加载器)   负责记载classpath中指定的jar包及目录中class 4)CustomClassLoader(自定义加载器)   属于应用程序根据自身需要自定义的ClassLoader,如tomcat、jboss都会根据j2ee规范自行实现。

(2)类加载器的顺序 1)加载过程中会先检查类是否被已加载,检查顺序是自底向上,从Custom ClassLoader到BootStrap ClassLoader逐层检查,只要某个classloader已加载就视为已加载此类,保证此类只所有ClassLoader加载一次。而加载的顺序是自顶向下,也就是由上层来逐层尝试加载此类。 2)在加载类时,每个类加载器会将加载任务上交给其父,如果其父找不到,再由自己去加载。

3)Bootstrap Loader(启动类加载器)是最顶级的类加载器了,其父加载器为null。

public class Demo {public static void main(String[] args) {C c = new C();}
}
class A{static {System.out.println("A static");}
}
​
class B extends A{static {System.out.println("B static");}
}
​
class D{static {System.out.println("D static");}
}
​
class C extends B{private static D d = new D();static {System.out.println("C static");}
}

打印

A static
B static
D static
C static

在上面的例子中,类C继承于B,B继承于A,C又依赖于D,在创建C时,会自动加载C继承的B和依赖的D,然后B又继承于A,先加载A,再加载B,再加载D,最后加载D

继承的优先级大于依赖

所有的变量初始化完,才会执行构造方法

在类的加载过程中,只有内部的变量创建完,才会去执行这个类的构造方法。

// 静态块 和 构造方法的加载顺序
public class ConstructorStep {public static void main(String[] args) {A2 a2 = new A2();}
}
​
class D2{static {System.out.println("D2 static");}public D2(){System.out.println("D2 constructor");}
}
​
class C2{static {System.out.println("C2 static");}public C2(){System.out.println("C2 constructor");}
}
​
class B2{C2 c2 = new C2();D2 d2 = new D2();static {System.out.println("B static");}public B2(){System.out.println("B2 constructor");}
}
​
class A2{B2 b2 = new B2();static {System.out.println("A static");}public A2(){System.out.println("A2 constructor");}
}

打印:

A static
B static
C2 static
C2 constructor
D2 static
D2 constructor
B2 constructor
A2 constructor
​

先创建A,执行A的静态块,再加载A的成员变量B2

执行B2的静态块,再加载B的成员变量C2,D2

执行C2的静态块,由于C2中没有成员变量需要加载,再执行C2的空参构造方法。

执行D2的静态块,同C2一样。

B2的成员变量加载完了,执行B2的构造方法

最后,A2同B2一样


总结

  1. 所有的类都会优先加载基类

  2. 静态成员的初始化优先

  3. 成员初始化完后,才会执行构造方法

  4. 静态成员的初始化与静态块的执行,发生在类加载的时候,静态成员的优先级大于静态块

  5. 类对象的创建以及静态块的访问,都会触发类的加载

执行顺序

静态成员变量 -> 静态代码块 -> 成员变量 -> 构造方法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/798683.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【JVM性能调优】- 合理分配堆空间

1、合理的堆空间该如何分配 Java内存各分区的大小对JVM的性能影响很大,不恰当的空间大小可能会埋下很多故障隐患,同时也会直接或间接影响JVM的提升速率、分配速率,所以如何将各分区调整到合适的大小就成了一个棘手的问题。大部分不具备线上JV…

Centos7下docker删除容器与镜像

个人记录 查看容器 docker ps -a停止容器运行 docker stop jenkins卸载容器 docker rm jenkins查看镜像 docker images卸载镜像 docker rmi IMAGE ID查看容器与镜像是否卸载完毕 docker images docker ps -a

avro c++编译与使用

一、arvo介绍 Avro 是 Hadoop 中的一个子项目,也是一个数据序列化系统,其数据最终以二进制格式,采用行式存储的方式进行存储。 Avro提供了: 1)、丰富的数据结构。 2)、可压缩、快速的二进制数据格式。 3)、一个用来存储持久化数据…

海外媒体宣发,穿透与世界的交流 - “保姆级”教程 - 大舍传媒

1. 引言 在当今高度信息化的世界,境外媒体宣发已经成为企业、品牌和政府机构推广自身形象、扩大影响力的重要手段。如何在国际舞台上有效传播信息,提高国际知名度,成为了许多组织面临的重要课题。大舍传媒凭借多年的境外媒体宣发经验&#x…

TYPE-C PD协议 OTG - 开启充电与数据传输

TYPE-C PD协议,作为一种先进的充电与数据传输协议,正以其卓越的性能引领着充电与数据传输技术的发展。它通过USB Type-C接口,实现了充电与数据传输的完美结合,为用户带来了前所未有的便捷体验。 TYPE-C PD协议的一大亮点在于其支…

WPF程序添加托盘图标

程序添加托盘图标 UI层 //添加handycontrol的引用xmlns:hc"https://handyorg.github.io/handycontrol"//添加NotifyIcon图标 实现单击 双击 二级菜单点击功能<hc:NotifyIconText"通知"Token"Info"><hc:NotifyIcon.ContextMenu><…

Python Flask:TypeError: unsupported operand type(s) for -: ‘str‘ and ‘int‘

背景 项目中常见的分页功能&#xff0c;代码如下&#xff1a; # 收藏列表 blog_base_blueprint.route(/resource_like_list/<int:user_id>, methods[POST]) def getResourceLikeList(user_id):page_size request.form.get(page_size, default2)page_num request.form…

opencv的kcf调参

cv::TrackerKCF::Params 是 OpenCV 中 KCF&#xff08;Kernelized Correlation Filter&#xff09;跟踪器的参数结构体&#xff0c;用于配置和优化跟踪器的行为。调整这些参数可以帮助你根据特定的应用场景和需求改进跟踪性能。以下是一些主要参数及其调整方法的详细说明&#…

Go 项目依赖注入wire工具最佳实践介绍与使用

文章目录 一、引入二、控制反转与依赖注入三、为什么需要依赖注入工具3.1 示例3.2 依赖注入写法与非依赖注入写法 四、wire 工具介绍与安装4.1 wire 基本介绍4.2 安装 五、Wire 的基本使用5.1 前置代码准备5.2 使用 Wire 工具生成代码 六、Wire 核心技术5.1 抽象语法树分析5.2 …

Docker【1】:Docker制作Oracle19C镜像

Docker【1】&#xff1a;Docker制作Oracle19C镜像 1、参考官方文档2、下载相关文件2.1、工具包2.2、Oracle安装包 3、制作镜像3.1、拷贝下载的oracle安装包到制作工具对应版本目录下3.2、开始制作镜像包3.3、制作完成 4、导出导入镜像4.1、镜像导出4.2、镜像导入 5、运行Oracle…

KNN课堂(分类课堂(可用kd树/特征归一化提高精度)))

实验代码&#xff1a; # 导入所需要的库 import numpy as np import pandas as pd from sklearn.ensemble import RandomForestClassifier from sklearn.model_selection import train_test_split from sklearn.neighbors import KNeighborsClassifier # 导入数据集 df pd.…

Node.js HTTP/2 CONTINUATION 拒绝服务漏洞(CVE-2024-27983)

Node.js 是开源、跨平台的 JavaScript 运行时环境。CONTINUATION泛洪攻击被发现存在于多个HTTP/2协议实现中。 在受影响版本中&#xff0c;由于Node.js针对HTTP/2协议的实现不当&#xff0c;未正确处理多个CONTINUATION帧的情况&#xff0c;在node::http2::Http2Session::~Htt…

缓存雪崩以及解决思路

缓存雪崩&#xff1a;缓存雪崩是指在同一时段大量的缓存key同时失效或者Redis服务宕机&#xff0c;导致大量请求到达数据库&#xff0c;带来巨大压力。 解决方案&#xff1a; 给不同的Key的TTL添加随机值 利用Redis集群提高服务的可用性 给缓存业务添加降级限流策略 给业务…

Centos7源码方式安装Elasticsearch 7.10.2单机版

下载 任选一种方式下载 官网7.10.2版本下载地址&#xff1a; https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch-7.10.2-linux-x86_64.tar.gz 网盘下载链接 链接&#xff1a;https://pan.baidu.com/s/1EJvUPGVOkosRO2PUaKibaA?pwdbnqi 提取码&#x…

C++ //练习 11.21 假定word_count是一个string到size_t的map,word是一个string,解释下面循环的作用:

C Primer&#xff08;第5版&#xff09; 练习 11.21 练习 11.21 假定word_count是一个string到size_t的map&#xff0c;word是一个string&#xff0c;解释下面循环的作用&#xff1a; while(cin>>word)word_count.insert({word, 0}).first->second;环境&#xff1a;…

opencv直线拟合+直线与图像交点坐标

opencv直线拟合直线与图像交点坐标 背景函数说明fitLineclipLine 代码 背景 在车道线拟合过程中&#xff0c;需要计算拟合直线与图像边界的交点&#xff0c;以确定车道区域。主要使用的函数fitLine和clipLine。 函数说明 fitLine /* 返回的lineParam信息如下&#xff1a; *…

如何在Linux中安装软件

文章目录 一、Linux应用程序基础1.Linux软件安装包分类2.应用程序和系统命令的关系3.常见的软件包的封装类型 二、安装软件的方式1.RPM包管理工具2.yum安装3.编译 一、Linux应用程序基础 1.Linux软件安装包分类 Linux源码包&#xff1a; 实际上&#xff0c;源码包就是一大堆源…

目标检测YOLO实战应用案例100讲-基于多特征融合的SSD目标检测

目录 前言 目标检测的国内外研究现状 传统的目标检测方法 深度学习的目标检测方法

Vue2 —— 学习(二)

目录 一、事件处理 &#xff08;一&#xff09;基本语法 &#xff08;二&#xff09;点击事件语法 1.事件绑定 2.回调函数 3.参数 event 注意&#xff1a; &#xff08;三&#xff09;事件修饰符 1.阻止默认事件 常用 2.阻止事件冒泡 常用 3.事件只触发一次 常用 …

深入理解GO语言——GC垃圾回收二

文章目录 前言一、Go V1.5的三色并发标记法总结 前言 书接上回&#xff0c;无论怎么优化&#xff0c;Go V1.3都面临这个一个重要问题&#xff0c;就是mark-and-sweep 算法会暂停整个程序 。 Go是如何面对并这个问题的呢&#xff1f;接下来G V1.5版本 就用 三色并发标记法 来优…