Java中类加载机制与反射

主要内容

详细内容

概述

JVM和类

当调用Java命令运行某个Java程序时候,该命令会启动一个Java虚拟机进程,同一个JVM的所有线程、所有变量都处于同一个进程里面,它们都使用该JVM进程的内存区,当系统出现以下几种情况的时候,JVM进程将会被终止。

(1)程序运行到最后正常结束;

(2)程序运行到使用System.exit(0)或者Runtime.getRuntime().exit()代码处结束进程;

(3)程序执行过程中遇到未捕获的异常或错误而结束;

(4)程序所在平台强制结束了JVM进程。

类的加载

当程序主动使用某个类的时候,如果该类还未被加载到内存当中,则系统会通过加载、连接、初始化这三个步骤来对该类进行初始化。如果没有以外,JVM会连续完成这3个步骤;可以将这三个步骤称之为类的加载。

类加载是指将类的class文件读入内存,并为之创建一个java.lang.Class对象,即,当程序中使用任何类时,系统都会为之建立一个java.lang.Class对象。

类的连接

当类被加载之后,系统会为之生成一个对应的Class对象,接着会进入连接阶段,连接阶段会负责把类的二进制数据合并到JRE中,类的连接可以分为三个阶段:

(1)验证:验证阶段用于检验被加载的类是否有正确的内部结构,并和其他类协调一致;

(2)准备:来准备阶段负责为类的静态属性分配内存,并设置为默认初始值;

(3)解析:将类的二进制数据中的符号引用替换成直接引用。

类的初始化

在类的初始化阶段,虚拟机负责对类进行初始化,主要就是对静态属性进行初始化。在Java类中对静态属性指定初始值有两种方式:

(1)声明静态属性时指定初始值;

(2)使用静态代码块为静态属性指定初始值,JVM会按这些语句在程序中的排列顺序依次执行它们。

JVM初始化一个类包含如下几个步骤:

(1)假如这个类还没有被加载和连接,则程序先加载并连接该类;

(2)假如该类的直接父类还没有被初始化,则先初始化其直接父类;

(3)假如类中有初始化语句,则系统依次执行这些初始化语句。

类的初始化时机

Java程序首次通过下面6种方式来使用某个类或接口时候,系统就会初始化该类或者接口。

(1)创建类的实例,为某个类创建实例的方式包括:使用new关键字来创建实例,通过反射来创建实例,通过反序列化的方式创建实例;

(2)调用某个类的静态方法;

(3)访问某个类或接口的静态属性,或者为该静态属性赋值;

(4)使用反射方式来强制创建某个类或接口对应的java.lang.Class对象。

(5)初始化某个类的子类;当初始化某个类的子类时,该子类的所有父类都会被初始化;

(6)直接使用java.exe命令来运行某个主类,当运行某个主类时,程序会先初始化该主类。

类加载器

类加载器简介

类加载器负责将.class文件加载到内存中,并为之生成对应的java.lang.Class对象。

JVM启动时,会形成由3个类加载器组成的初始类加载器层次结构。

(1)Bootstrap ClassLoader:根类加载器;它负责加载Java的核心类;它并不是java.lang.ClassLoader的子类,而是由JVM自身实现的,下面程序可以实现获取根类加载器所加在的核心类库。

public class BootstrapTest
{
	public static void main(String[] args)
	{
		// 获取根类加载器所加载的全部URL数组
		URL[] urls = sun.misc.Launcher.getBootstrapClassPath().getURLs();
		// 遍历、输出根类加载器加载的全部URL
		for (int i = 0; i < urls.length; i++)
		{
			System.out.println(urls[i].toExternalForm());
		}
	}
}

运行的结果如下所示:

(2)Extension ClassLoader:扩展类加载器;它负责加载JRE的扩展陌路中的JAR包中的类。

(3)System ClassLoader:系统类加载器;它负责在JVM启动时加载来自java命令的-classpath选项、java.class.path系统属性,或者CLASSPATH环境变量所制定的JAR包和类路径。

类加载机制

JVM的类加载机制主要有如下三种机制:

(1)全盘负责:指当一个类加载器负责加载某个Class时,该Class所依赖的和引用的其它Class也将有该类加载器负责载入,除非显式使用另外一个类加载器来载入;

(2)父类委托:指先让父类加载器视图加载该Class,只有在父类加载器无法加载该类时才尝试从自己的类路径中加载该类;

(3)缓存机制:指所有加载过的Class都会被缓存,当程序中需要使用某个Class时候,类加载器先从缓存区中搜寻该Class,只有当缓存去中不存在该Class对象时候,系统才会读取该类对应的二进制数据,并将其转换成Class对象,存入缓存区中。这就是为什么修改了Class后,必须重新启动JVM,程序所做的修改才会生效的原因。

以下代码演示访问JVM的类加载器。

public class ClassLoaderPropTest
{
	public static void main(String[] args)
	{
		// 获取系统类加载器
		ClassLoader systemLoader = ClassLoader.getSystemClassLoader();
		System.out.println("系统类加载器为: " + systemLoader);
		// 获取系统类加载器的父类加载期,得到扩展类加载器
		ClassLoader extensionLoader = systemLoader.getParent();
		System.out.println("扩展类加载器: " + extensionLoader);
		System.out.println("扩展类加载器的加载路径:" + System.getProperty("java.ext.dirs"));
		System.out.println("扩展类加载器的parent: " + extensionLoader.getParent());
	}

}

运行的结果如下所示:

类加载器加载Class大致要经过如下8个步骤,

(1)检测此Class是否载入过(即在缓存区中是否有此Class),有则直接进入第8步(红线表示),否则直接进入第2步;

(2)如果父类加载器不存在(如果没有父类加载器,则要么parent一定是根类加载器,要么本身就是根类加载器),则调到第

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值