关于jvm的三个类加载器以及jar包的生成

本文介绍Java中动态代理的实现原理,包括核心类Proxy和InvocationHandler的作用,以及类加载器的不同类型和它们之间的关系。通过代码示例展示了如何确定类是由哪个类加载器加载的。
package com.cgm.invokeclass;

import java.lang.reflect.Method;
/*
* 关于动态代理的两个核心类
* 1.Proxy
* 用来创建给定接口的子类,在内存中动态创建 $proxy() 执行一次
* 2.InvicationHandler 执行句柄。。在执行时可以获取被代理类的反射。 执行多次,用户的每一次调用都会被这个句柄所拦截
*
* 3.关于类加载器 有3
* 1 BootStrapClassLoader 根类加载器 ----加载jdk下rt.jar
* 2 ExtClassloader java扩展类加载器---- jdk下 的 ext文件夹下了类
* 3 AppClassLoader----加载系统的类 我们自己写的类 Web_info下 class文件夹下的类
* 因为jdk的父类委托机制 任务由下往上 委托执行 下列代码证明
*
* 如何用cmd给class打jar包
* 如把包 com/cgm/ss/VBBB.class
* copy到某个目录下 D:a目录----》
* 开cmd 到 D:\a\com>cd..
* D:\a>.jar -vcf aa.jar com.*
*
*/
public class ProxyDemo {
public static void main(String[] args) throws Exception {
ProxyDemo p=new ProxyDemo();
System.out.println(p.getClass().getClassLoader());
System.out.println(ProxyDemo.class.getClassLoader()); //和上面同一个内存地址 应该是 由AppClassLoader加载
System.out.println(ProxyDemo.class.getClassLoader().getParent());//ExtClassLoader 父类
System.out.println(ProxyDemo.class.getClassLoader().getParent().getParent()); //返回null,跟类加载器不允许使用
System.out.println(String.class.getClassLoader()); //取得根类加载器 因为String是jre下rt.jar的方法
}
@Override
public String toString() {

return "你好朋友";
}
}
JVM 类加载器Java 运行时环境的重要组成部分,负责将类的字节码文件加载到 JVM 中并转换为可执行的类对象。其工作原理和作用在 Java 程序的运行过程中具有核心意义。 ### 类加载器的工作原理 JVM 类加载器采用 **双亲委派机制**(Parent Delegation Model),这是一种层次化的类加载机制。当一个类加载器收到类加载请求时,它不会立即加载该类,而是先将请求委托给其父类加载器进行处理。只有当父类加载器无法加载该类时,子类加载器才会尝试自己加载。这种机制确保了类加载的统一性和安全性,避免了类的重复加载和冲突问题[^1]。 类加载过程主要括以下三个阶段: 1. **加载**(Loading):查找并加载类的 `.class` 文件,将其转换为 JVM 可识别的二进制字节流。 2. **链接**(Linking):括验证、准备和解析三个子阶段,确保类的结构正确性,并为类变量分配内存和设置初始值。 3. **初始化**(Initialization):执行类构造器 `<clinit>` 方法,对类的静态变量和静态代码块进行初始化[^1]。 ### 类加载器的类型及其作用 JVM 中存在以下主要的类加载器: #### 启动类加载器(Bootstrap ClassLoader) - 是 JVM 自带的类加载器,由 C++ 实现,不继承 `java.lang.ClassLoader`。 - 负责加载 Java 核心类库(如 `rt.jar` 中的类),例如 `java.lang.String` 和 `java.lang.Object`。 - 在 Java 代码中无法直接获取该类加载器,调用 `ClassLoader.getSystemClassLoader()` 会返回 `null`[^2]。 #### 扩展类加载器(Extension ClassLoader) - 由 Java 编写,继承自 `ClassLoader`。 - 负责加载 JRE 的扩展类库,通常位于 `jre/lib/ext` 目录或由 `java.ext.dirs` 系统属性指定的位置。 - 是应用程序类加载器的父加载器[^3]。 #### 应用程序类加载器(Application ClassLoader) - 也称为系统类加载器,由 Java 编写。 - 负责加载用户类路径(ClassPath)上的类,括通过 `-classpath` 或 `-cp` 指定的类路径。 - 通常作为程序中默认的类加载器,可以通过 `ClassLoader.getSystemClassLoader()` 获取其实例[^4]。 ### 类加载器的作用 1. **类的加载**:将类的 `.class` 文件加载到 JVM 中,生成对应的 `Class` 对象。 2. **命名空间隔离**:不同类加载器加载的类被视为不同的类,即使它们的类名和字节码完全相同。这种机制有助于实现模块化和隔离性。 3. **安全性保障**:通过双亲委派机制和类加载器的隔离特性,JVM 能够防止恶意代码篡改核心类库。 4. **动态加载**:支持运行时动态加载类,为插件系统、热部署等高级特性提供基础支持[^5]。 ### 示例代码:获取类加载器 ```java public class ClassLoaderDemo { public static void main(String[] args) { // 获取系统类加载器 ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader(); System.out.println("系统类加载器: " + systemClassLoader); // 获取 String 类的类加载器(启动类加载器) ClassLoader stringClassLoader = String.class.getClassLoader(); System.out.println("String 类的类加载器: " + stringClassLoader); // 输出 null // 获取当前类的类加载器(系统类加载器) ClassLoader currentClassLoader = ClassLoaderDemo.class.getClassLoader(); System.out.println("当前类的类加载器: " + currentClassLoader); } } ``` ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值