浅谈Java中类加载机制

首先来了解一下jvm(java虚拟机)中的几个比较重要的内存区域,这几个区域在java类的生命周期中扮演着比较重要的角色:

方法区:在java的虚拟机中有一块专门用来存放已经加载的类信息、常量、静态变量以及方法代码的内存区域,叫做方法区。

常量池:常量池是方法区的一部分,主要用来存放常量和类中的符号引用等信息。

堆区:用于存放类的对象实例。

栈区:也叫java虚拟机栈,是由一个一个的栈帧组成的后进先出的栈式结构,栈桢中存放方法运行时产生的局部变量、方法出口等信息。当调用一个方法时,虚拟机栈中就会创建一个栈帧存放这些数据,当方法调用完成时,栈帧消失,如果方法中调用了其他方法,则继续在栈顶创建新的栈桢。

除了以上四个内存区域之外,jvm中的运行时内存区域还包括本地方法栈和程序计数器,这两个区域与java类的生命周期关系不是很大,在这里就不说了,感兴趣的朋友可以自己百度一下。

类的生命周期

当我们编写一个java的源文件后,经过编译会生成一个后缀名为class的文件,这种文件叫做字节码文件,只有这种字节码文件才能够在java虚拟机中运行,java类的生命周期就是指一个class文件从加载到卸载的全过程。

一个java类的完整的生命周期会经历加载、连接、初始化、使用、和卸载五个阶段,当然也有在加载或者连接之后没有被初始化就直接被使用的情况,如图所示:

下面我们就依次来说一说这五个阶段。

加载

在java中,我们经常会接触到一个词——类加载,它和这里的加载并不是一回事,通常我们说类加载指的是类的生命周期中加载、连接、初始化三个阶段。在加载阶段,java虚拟机会做什么工作呢?其实很简单,就是找到需要加载的类并把类的信息加载到jvm的方法区中,然后在堆区中实例化一个java.lang.Class对象,作为方法区中这个类的信息的入口。

类的加载方式比较灵活,我们最常用的加载方式有两种,一种是根据类的全路径名找到相应的class文件,然后从class文件中读取文件内容;另一种是从jar文件中读取。另外,还有下面几种方式也比较常用:

从网络中获取:比如10年前十分流行的Applet。

根据一定的规则实时生成,比如设计模式中的动态代理模式,就是根据相应的类自动生成它的代理类。

从非class文件中获取,其实这与直接从class文件中获取的方式本质上是一样的,这些非class文件在jvm中运行之前会被转换为可被jvm所识别的字节码文件。

对于加载的时机,各个虚拟机的做法并不一样,但是有一个原则,就是当jvm“预期”到一个类将要被使用时,就会在使用它之前对这个类进行加载。比如说,在一段代码中出现了一个类的名字,jvm在执行这段代码之前并不能确定这个类是否会被使用到,于是,有些jvm会在执行前就加载这个类,而有些则在真正需要用的时候才会去加载它,这取决于具体的jvm实现。我们常用的hotspot虚拟机是采用的后者,就是说当真正用到一个类的时候才对它进行加载。

加载阶段是类的生命周期中的第一个阶段,加载阶段之后,是连接阶段。有一点需要注意,就是有时连接阶段并不会等加载阶段完全完成之后才开始,而是交叉进行,可能一个类只加载了一部分之后,连接阶段就已经开始了。但是这两个阶段总的开始时间和完成时间总是固定的:加载阶段总是在连接阶段之前开始,连接阶段总是在加载阶段完成之后完成。

连接

连接阶段比较复杂,一般会跟加载阶段和初始化阶段交叉进行,这个阶段的主要任务就是做一些加载后的验证工作以及一些初始化前的准备工作,可以细分为三个步骤:验证、准备和解析。

验证:当一个类被加载之后,必须要验证一下这个类是否合法,比如这个类是不是符合字节码的格式、变量与方法是不是有重复、数据类型是不是有效、继承与实现是否合乎标准等等。总之,这个阶段的目的就是保证加载的类是能够被jvm所运行。

准备:准备阶段的工作就是为类的静态变量分配内存并设为jvm默认的初值,对于非静态的变量,则不会为它们分配内存。有一点需要注意,这时候,静态变量的初值为jvm默认的初值,而不是我们在程序中设定的初值。jvm默认的初值是这样的:

基本类型(int、long、short、char、byte、boolean、float、double)的默认值为0。

引用类型的默认值为null。

常量的默认值为我们程序中设定的值,比如我们在程序中定义final static int a = 100,则准备阶段中a的初值就是100。

解析:这一阶段的任务就是把常量池中的符号引用转换为直接引用。那么什么是符号引用,什么又是直接引用呢?

我们来举个例子:我们要找一个人,我们现有的信息是这个人的身份证号是1234567890。只有这个信息我们显然找不到这个人,但是通过公安局的身份系统,我们输入1234567890这个号之后,就会得到它的全部信息:比如安徽省黄山市余暇村18号张三,通过这个信息我们就能找到这个人了。这里,123456790就好比是一个符号引用,而安徽省黄山市余暇村18号张三就是直接引用。

在内存中也是一样,比如我们要在内存中找一个类里面的一个叫做show的方法,显然是找不到。但是在解析阶段,jvm就会把show这个名字转换为指向方法区的的一块内存地址,比如c17164,通过c17164就可以找到show这个方法具体分配在内存的哪一个区域了。这里show就是符号引用,而c17164就是直接引用。在解析阶段,jvm会将所有的类或接口名、字段名、方法名转换为具体的内存地址。

连接阶段完成之后会根据使用的情况(直接引用还是被动引用)来选择是否对类进行初始化。

初始化

如果一个类被直接引用,就会触发类的初始化。在java中,直接引用的情况有:

通过new关键字实例化对象、读取或设置类的静态变量、调用类的静态方法。

通过反射方式执行以上三种行为。

初始化子类的时候,会触发父类的初始化。

作为程序入口直接运行时(也就是直接调用main方法)。

除了以上四种情况,其他使用类的方式叫做被动引用,而被动引用不会触发类的初始化。请看主动引用的示例代码:

import java.lang.reflect.Field;

import java.lang.reflect.Method;

class InitClass{

static {

System.out.println("初始化InitClass");

}

public static String a = null;

public static void method(){}

}

class SubInitClass extends InitClass{}

public class Test1 {

/**

* 主动引用引起类的初始化的第四种情况就是运行Test1的main方法时

* 导致Test1初始化,这一点很好理解,就不特别演示了。

* 本代码演示了前三种情况,以下代码都会引起InitClass的初始化,

* 但由于初始化只会进行一次,运行时请将注解去掉,依次运行查看结果。

* @param args

* @throws Exception

*/

public static void main(String[] args) throws Exception{

// 主动引用引起类的初始化一: new对象、读取或设置类的静态变量、调用类的静态方法。

// new InitClass();

// InitClass.a = "";

// String a = InitClass.a;

// InitClass.method();

// 主动引用引起类的初始化二:通过反射实例化对象、读取或设置类的静态变量、调用类的静态方法。

// Class cls = InitClass.class;

// cls.newInstance();

// Field f = cls.getDeclaredField("a");

// f.get(null);

// f.set(null, "s");

// Method md = cls.getDeclaredMethod("method");

// md.invoke(null, null);

// 主动引用引起类的初始化三:实例化子类,引起父类初始化。

// new SubInitClass();

}

}

上面的程序演示了主动引用触发类的初始化的四种情况。

类的初始化过程是这样的:按照顺序自上而下运行类中的变量赋值语句和静态语句,如果有父类,则首先按照顺序运行父类中的变量赋值语句和静态语句。先看一个例子,首先建两个类用来显示赋值操作:

public class Field1{

public Field1(){

System.out.println("Field1构造方法");

}

}

public class Field2{

public Field2(){

System.out.println("Field2构造方法");

}

}

下面是演示初始化顺序的代码:

class InitClass2{

static{

System.out.println("运行父类静态代码");

}

public static Field1 f1 = new Field1();

public static Field1 f2;

}

class SubInitClass2 extends InitClass2{

static{

System.out.println("运行子类静态代码");

}

public static Field2 f2 = new Field2();

}

public class Test2 {

public static void main(String[] args) throws ClassNotFoundException{

new SubInitClass2();

}

}

上面的代码中,初始化的顺序是:第03行,第05行,第11行,第13行。第04行是声明操作,没有赋值,所以不会被运行。而下面的代码:

class InitClass2{

public static Field1 f1 = new Field1();

public static Field1 f2;

static{

System.out.println("运行父类静态代码");

}

}

class SubInitClass2 extends InitClass2{

public static Field2 f2 = new Field2();

static{

System.out.println("运行子类静态代码");

}

}

public class Test2 {

public static void main(String[] args) throws ClassNotFoundException{

new SubInitClass2();

}

}

初始化顺序为:第02行、第05行、第10行、第12行,各位可以运行程序查看结果。

在类的初始化阶段,只会初始化与类相关的静态赋值语句和静态语句,也就是有static关键字修饰的信息,而没有static修饰的赋值语句和执行语句在实例化对象的时候才会运行。

使用

类的使用包括主动引用和被动引用,主动引用在初始化的章节中已经说过了,下面我们主要来说一下被动引用:

引用父类的静态字段,只会引起父类的初始化,而不会引起子类的初始化。

定义类数组,不会引起类的初始化。

引用类的常量,不会引起类的初始化。

被动引用的示例代码:

class InitClass{

static {

System.out.println("初始化InitClass");

}

public static String a = null;

public final static String b = "b";

public static void method(){}

}

class SubInitClass extends InitClass{

static {

System.out.println("初始化SubInitClass");

}

}

public class Test4 {

public static void main(String[] args) throws Exception{

// String a = SubInitClass.a;// 引用父类的静态字段,只会引起父类初始化,而不会引起子类的初始化

// String b = InitClass.b;// 使用类的常量不会引起类的初始化

SubInitClass[] sc = new SubInitClass[10];// 定义类数组不会引起类的初始化

}

}

最后总结一下使用阶段:使用阶段包括主动引用和被动引用,主动饮用会引起类的初始化,而被动引用不会引起类的初始化。

当使用阶段完成之后,java类就进入了卸载阶段。

卸载

在类使用完之后,如果满足下面的情况,类就会被卸载:

该类所有的实例都已经被回收,也就是java堆中不存在该类的任何实例。

加载该类的ClassLoader已经被回收。

该类对应的java.lang.Class对象没有任何地方被引用,无法在任何地方通过反射访问该类的方法。

如果以上三个条件全部满足,jvm就会在方法区垃圾回收的时候对类进行卸载,类的卸载过程其实就是在方法区中清空类信息,java类的整个生命周期就结束了。

为了让学习变得轻松、高效,今天给大家免费分享一套Java教学资源。帮助大家在成为Java架构师的道路上披荆斩棘。需要资料的欢迎加入学习交流群:9285,05736

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

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

相关文章

Java学习四步曲,助你成长!

对于很多新手来说,可能JAVA的学习会很难。这种难度体现在语言的专业性、学习的不系统、条理的不清晰以及缺乏足够的耐心。实际上想要学习好JAVA,除了有足够的心理准备,还要有挑战JAVA终极四部曲的信心和勇气,那么四部曲是什么&…

Java里的 for (;;) 与 while (true),哪个更快?

在JDK8u的jdk项目下做个很粗略的搜索: 并没有差多少。 其次,for (;;) 在Java中的来源。个人看法是喜欢用这种写法的人,追根溯源是受到C语言里的写法的影响。这些人不一定是自己以前写C习惯了这样写,而可能是间接受以前写C的老师、…

Java中的null到底是什么?

让我们从下面的陈述开始: String xnull; 1. 这句话到底是什么意思? 回想一下什么是变量,什么是值。 一个常见的比喻是变量类似于一个盒子。 就像您可以使用一个框来存储某些东西一样,您也可以使用一个变量来存储一个值。 在声明变量时…

学习Java编程-Java Timezone类常见问题

今天遇到了一个比较有意思的问题,从服务器上封装好的java.sql.timestamp对象返回到本地客户端程序后与数据库中的时间相差了整整14个小时。因为跟客户的时差是14个小时,所以大体怀疑是时差问题。 所以计划在客户端程序执行之前首先设置默认的TimeZone: …

热门专业学习之关于java的一些知识

1. JAVASE 首先要学 JavaSE,这是毋庸置疑的。与此同时,和 JavaSE 的学习同步,建议大家研究一下数据结构与算法。 在 JavaSE 完成之后,可以试着完成一些小项目,同时关注一下设计模式的内容,不必强求自己能…

企业需求的Java程序员是什么样子的

选择学习Java编程​语言,大部分人还是冲着高薪就业去的,既然如此,就业是学习Java的最终目的,企业需要什么我们就学什么。 下面分析一下企业需要什么,这些也是我们在学习中应该着重关心的。当然,如果你是因…

如何提高Java代码的可重用性?

提高java代码可重用性有哪些方法措施,以下就讲解了三种关于提高java代码可重用性的措施,一起来了解一下吧~ 改写类的实例方法 通过类继承实现代码重用不是精确的代码重用技术,因此它并不是最理想的代码重用机制。继承总是带来一些多余的方法…

Java常见面试题之类的加载过程

程序员看似光鲜的就业前景面前,逃不过的是层层的面试,想要进前沿的大公司没有个五六七八面,是不可能滴!而找工作的首个关卡就是笔试,想要获得高薪工作的小伙伴,先刷一波面试题吧! 类加载过程主…

作为Java程序员,这些开源工具你应该要学习!

1. JIRA Atlassian的JIRA是当前敏捷开发领域最重要的工具之一。它用于错误跟踪,问题跟踪和项目管理。如果你遵循敏捷开发方法,例如Sprint和Scrum,那么你必须了解JIRA。它允许您创建Spring循环并跟踪软件开发的进度。 JIRA 是目前比较流行的基…

在Java编码中,如何减少bug数量

众所周知,Java编程语言在IT行业是企业中不可缺少的。不管,从Web应用到Android应用,这款语言已经被广泛用于开发各类应用及代码中的复杂功能。但在编写代码时,bug永远是困扰每一位从业者的头号大难题。今天就与大家分享几个关于减少…

为什么要学习Java EE?需要掌握哪些技能?

随着互联网的不断发展,Java作为一种古老的编程语言,全年仍占据着编程语言的榜首。那么编辑应该如何学习Java呢? 选择学习JavaEE或JavaME(或者你想继续学习Java SE的深度,只要你喜欢,你就可以一直深入下去&…

Java Socket

什么是Socket Socket的概念很简单,它是网络上运行的两个程序间双向通讯的一端,既可以接收请求,也可以发送请求,利用它可以较为方便地编写网络上数据的传递。 所以简而言之,Socket就是进程通信的端点,Sock…

40个Java 多线程问题总结

1、多线程有什么用? 一个可能在很多人看来很扯淡的一个问题:我会用多线程就好了,还管它有什么用?在我看来,这个回答更扯淡。所谓"知其然知其所以然","会用"只是"知其然"&am…

学好Java的八个条件

世界上没有什么捷径可以成为大师,但老师告诉我们学习Java的一些基本原则是可以遵循的。接下来,前锋广州老师分享了学习Java编程的8个条件。你有多少? 1。坚实基础 数据结构、离散数学和编译原理是所有计算机科学的基础。如果我们不掌握它们&…

Java面向对象基础接口和抽象的理解

第一,抽象类: 我们都知道,在面向对象的领域中,一切都是一个对象,所有的对象都是用类来描述的,但不是所有的类都是用对象来描述的。如果一个类没有足够的信息来描述一个特定的对象,并且需要其他…

java中容易被忽视的基本概念

概念1: try-catch-finally块中,finally块在以下几种情况将不会执行。 (1)finally块中发生了异常。 (2)程序所在线程死亡。 (3)在前面的代码中用了System.exit(&#…

JAVA四种遍历Map的方法

导入java.util.hashmap&#xff1b; 导入java.util.iterator&#xff1b; 导入java.util.map&#xff1b; 导入java.util.set&#xff1b; 公共类映射{ 公共静态void main&#xff08;string[]args&#xff09;{ mapmapnew hashmap<>&#xff08;&#xff09;&…

spring中基于Java容器配置注解的区别及使用场景

转载自百家号作者&#xff1a;有趣的代码 Component、Service、Controller、Repository&#xff1b;Configuration、Bean区别&#xff0c;使用场景 从spring3.0开始&#xff0c;基于javaconfig的项目&#xff0c;支持使用java来定义bean&#xff0c;而不是传统的xml文件。Comp…

Java的文件流操作

文件系统 FileSystem类的对象表示Java程序中的文件系统。 FileSystem对象用于执行两个任务&#xff1a; Java程序和文件系统之间的接口。 一个工厂用于创建许多类型的文件系统相关对象和服务。 FileSystem对象与平台相关。 创建文件系统 要获取默认的FileSystem对象&…

学习Java编程,英语对我们来说有多重要?

Java软件开发需要学英语吗&#xff1f;学软件开发大家都知道&#xff0c;动手写代码远远比那些个只看理论知识要强得多。很多人因为不懂软件开发流程&#xff0c;就会觉得软件开发肯定很难&#xff0c;实际上不然&#xff0c;软件开发并没有想象中的那么难&#xff0c;也完全没…