JAVA类的加载(4) ——类之间能够隔离&类占用的资源能回收
一、类加载体系
类加载方式:代理模式 或 双亲委托
例1:
1 package classloader.system;
2
3 public class Example {
4 public static void main(String[] args) {
5 /*应用的类加载器是AppClassLoader,首先委托父ClassLoder(ExtClassLoder)从他自己的资源池中(jre/lib/ext)找这个类,
6 * 在这之前,ExtClassLoder又委托父ClassLoader(BootStrapClassLoader)从jre/lib找这个类;
7 * BootStrapClassLoader在jre/lib没找到需要的类,返回到ExtClassLoder;ExtClassLoder在jre/lib/ext没找到需要的类,返回到AppClassLoader
8 *AppClassLoader从自己的资源池中查找 c = findClass(name);
9 *
10 */
11 ClassLoader cl = Example.class.getClassLoader();
12 while (cl != null) {
13 System.out.println(cl.toString());
14 cl = cl.getParent();
15 }
16 }
17 }
结果:
sun.misc.Launcher$AppClassLoader@425224ee
sun.misc.Launcher$ExtClassLoader@1ef6a746
例2:在eclipse中点击例1中的11行ClassLoader,进入ClassLoader类的代码,找到loadClass方法,里面就是详细的类加载代码
1 protected synchronized Class<?> loadClass(String name, boolean resolve)
2 throws ClassNotFoundException
3 {
4 // First, check if the class has already been loaded
5 Class c = findLoadedClass(name);
6 if (c == null) {
7 try {
8 if (parent != null) {
9 c = parent.loadClass(name, false); //递归通过父加载器加载
10 } else {
11 c = findBootstrapClassOrNull(name); //从根加载器的资源池中查找这个类
12 }
13 } catch (ClassNotFoundException e) {
14 // ClassNotFoundException thrown if class not found
15 // from the non-null parent class loader
16 }
17 if (c == null) {
18 // If still not found, then invoke findClass in order
19 // to find the class.
20 c = findClass(name); //应用加载器从自己的资源池中查找这个类
21 }
22 }
23 if (resolve) {
24 resolveClass(c);
25 }
26 return c;
27 }
1 package classloader.system2;
2
3 import java.lang.reflect.Method;
4 import java.net.MalformedURLException;
5 import java.net.URL;
6 import java.net.URLClassLoader;
7 import java.io.File;
8
9 public class Example {
10 public static void main(String[] args) throws Exception {
11 /*前面假设是一个web容器,web容器通过前面的方式启动起来了;当你扔一个war包进去,war包有自己的ClassLoader以及自己的url,url指向war包里面的lib目录
12 * 相当于可以从url里面加载类 用web应用的classloader去加载相关的类,并且调它的初始化方法,就可以让我自己的war包运行起来
13 * war包里面都有一个web.xml,web.xml描述了Servlet,相当于描述了全称类名叫什么,我就可以启动相关的Servlet;相关的Servlet启动以后,我就可以接受请求并进行分发,web容器就基本上运行起来了
14 *
15 *
16 */
17 URL url = new File(args[0]).toURL();
18 ClassLoader cl = new URLClassLoader(new URL[]{url});
19 Class<?> tvClass = cl.loadClass("classloader.system2.Television");
20 Object tv = tvClass.newInstance(); //因为我们当前classpath下是没有Television类的,所以这里定义为Object tv
21
22 Object panel = cl.loadClass("classloader.system2.Panel").newInstance(); //load一个面板并实例化
23 Method setPanelMethod = tvClass.getMethod("setPanel", Object.class);
24 setPanelMethod.invoke(tv, panel);
25 Method playVideoMethod = tvClass.getMethod("playVideo", new Class[]{});
26 playVideoMethod.invoke(tv, new Object[]{});
27
28 System.out.println("不同的ClassLoader加载。。。");
29 ClassLoader cl2 = new URLClassLoader(new URL[]{url});
30 Object panel2 = cl2.loadClass("classloader.system2.Panel").newInstance();
31 setPanelMethod.invoke(tv, panel2); //tv是cl加载的,panel2是cl2加载的,两个不同的classload,它们之间的对象是不能直接进行装配的,会认为不一样的
32 }
33 }
34
35 //将Panel和Televison两个类拷贝到特定目录并编译,并删除当前classpath下的两个类
//Television.java
1 package classloader.system2;
2
3 public class Television {
4 private Panel panel; //面板
5
6 public void playVideo() {
7 panel.display();
8 }
9
10 public void setPanel(Object panel) {
11 this.panel = (Panel)panel;
12 }
13 }
//Panel.java
1 package classloader.system2;
2
3 public class Panel {
4 public void display() {
5 System.out.println("In Panel: display()");
6 }
7 }
运行:
编译好后,将Panel和Televison两个类class文件拷贝到特定目录(如下),并删除当前classpath下的两个类(main的参数String[] args为d:\tmp)
1、ClassLoader cl = new URLClassLoader(new URL[]{url}); 这个classLoader所指向的url在d:\,d:\下Class<?> tvClass = cl.loadClass("classloader.system2.Television");时找不到,所以就报错了
2、填写正确的路径
classloader.system2.Panel cannot be cast to classloader.system2.Panel报错原因如下:
tv是cl加载的,panel2是cl2加载的,两个不同的classload,它们之间的对象是不能直接进行装配的,会认为不一样的
对象与类的加载 小结:
1、并非所有的类都需要在CLASSPATH中
2、对象的赋值与转换与其ClassLoader相关
3、类在一个ClassLoader中只加载一次
三、与类加载相关的异常
1、ClassNotFoundException
2、NoClassDefFoundError
1 package classloader.exception;
2
3
4 public class Example {
5 public static void main(String[] args) throws Exception {
6 Television tv = (Television) Example.class.getClassLoader()
7 .loadClass("classloader.exception.Television").newInstance(); //loadClass("classloader.exception.Television1") 就会抛java.lang.ClassNotFoundException:
8 //exception
9 tv.playVideo();
10 }
11 }
12
13 class Television {
14 //private Panel panel; 会报错at classloader.exception.Television.playVideo(Example.java:17)
15 private Panel panel = new Panel(); //Televison编译强依赖于Panel,如果编译好后再删除Panel.class,运行时会报java.lang.NoClassDefFoundError: classloader/exception/Panel
16 //error
17 public void playVideo() {
18 panel.display();
19 }
20
21 public void setPanel(Object panel) {
22 this.panel = (Panel)panel;
23 }
24 }
25
26
27 class Panel {
28 public void display() {
29 System.out.println("In Panel: display()");
30 }
31 }
四、类的回收
1、ClassLoader加载类,类实例化对象
2、当某个ClassLoader加载的所有类实例化的所有对象都被回收了,则该CL会被回收
五、类的热替换思路
参考:
JAVA类的加载(4) ——类之间能够隔离&类占用的资源能回收的更多相关文章
- JAVA类的加载、连接与初始化
JAVA类的加载.连接与初始化 类的声明周期总共分为5个步骤1.加载2.连接3.初始化4.使用5.卸载 当java程序需要某个类的时候,java虚拟机会确保这个类已经被加载.连接和初始化,而连接这个类 ...
- jvm系列(一):java类的加载机制
java类的加载机制 1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装 ...
- [Java]类的生命周期(上)类的加载和连接[转]
本文来自:曹胜欢博客专栏.转载请注明出处:http://blog.csdn.net/csh624366188 类加载器,顾名思义,类加载器(class loader)用来加载 Java 类到 Java ...
- java类生命周期,类的“加载,连接,初始化,使用,卸载过程”详解
“ 如果说核心类库的 API 比做数学公式的话,那么 Java 虚拟机的知识就好比公式的推导过程” 每本Java入门书籍在介绍Java这门语言的时候都会提到Java跨平台,“一次解释,到处运行的特点“ ...
- JVM(1):Java 类的加载机制
原文出处: 纯洁的微笑 java类的加载机制 1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang. ...
- Java 类的加载与初始化
本文结构: 1.先看几道题 2.类的加载于初始化 (1)类的加载 (2)类的初始化 (a)会发生类的初始化的情况 (b)不会发生类的初始化的情况 首先看几道题. 解析可在看完讲解后再看 Demo1 p ...
- <JVM中篇:字节码与类的加载篇>03-类的加载过程(类的生命周期)详解
笔记来源:尚硅谷JVM全套教程,百万播放,全网巅峰(宋红康详解java虚拟机) 同步更新:https://gitee.com/vectorx/NOTE_JVM https://codechina.cs ...
- JVM 1.类的加载、连接、初始化
Java类的加载是由类加载器来完成的,过程如下: 首先,加载是把硬盘.网络.数据库等的class文件中的二进制数据加载到内存的过程,然后会在Java虚拟机的运行时数据区的堆区创建一个Class对象,用 ...
- JVM-01:类的加载机制
本文从 纯洁的微笑的博客 转载 原地址:http://www.ityouknow.com/jvm.html 类的加载机制 1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内 ...
- 24.类的加载机制和反射.md
目录 1类的加载连接和初始化 1.1类的加载过程 1.2类的加载器 1.2.1类的加载机制 1类的加载连接和初始化 1.1类的加载过程 类的加载过程简单为分为三步:加载->连接->初始化 ...
随机推荐
- Redis 备忘录
redis是什么 Redis 是一个高性能的key-value数据库 常用操作 下载 官网:https://redis.io/ Linux版:https://redis.io/download Win ...
- 定义一个类,在实例化的时候,抛出NameError异常
代码1:class cla: def __init__(self): #raise NameError # 抛出异常 print(r) cla() 运行截图:
- Midjourney 创建私人画图机器人(保姆级教程)
本教程收集于:AIGC从入门到精通教程汇总 之前给大家介绍过了Midjourney 的注册教程:AI绘画:Midjourney 注册(保姆级教程) 也有Stable Diffusion(开源)的本地搭 ...
- Go 上下文的理解与使用
为什么需要 context 在 Go 程序中,特别是并发情况下,由于超时.取消等而引发的异常操作,往往需要及时的释放相应资源,正确的关闭 goroutine.防止协程不退出而导致内存泄露.如果没有 c ...
- Codeforces Round 882 div.2 A
Smiling&Weeping ----总有人间一两风,填我十万八千梦 A. The Man who became a God time limit per test 1 second mem ...
- DP模拟题
Smiling & Weeping ----寒灯纸上,梨花雨凉,我等风雪又一年 # [NOIP2007 普及组] 守望者的逃离 ## 题目背景 恶魔猎手尤迪安野心勃勃,他背叛了暗夜精灵,率领深 ...
- Fabric 2.x 智能合约开发记录
表象:Return schema invalid. required items must be unique [recovered] 虽然 Fabric v2.2 已经发布了很久了,但之前因为项目历 ...
- 图解 LeetCode 算法汇总——二分查找
二分查找(Binary Search)是一种在有序数组中查找特定元素的高效算法.它的基本思想是将目标值与数组中间的元素进行比较,如果目标值小于中间元素,则在数组的左半部分继续查找,否则在右半部分查找, ...
- .NET周刊【9月第4期 2023-09-24】
国内文章 有趣的"可扩展近似计数"算法 https://zhuanlan.zhihu.com/p/656817283 在编程的世界里看见数学的身影,会让我充满好奇和兴奋.这不,在一 ...
- MySQL系列之主从复制基础——企业高可用性标准、主从复制简介、主从复制前提(搭建主从的过程)、主从复制搭建、主从复制的原理、主从故障监控\分析\处理、主从延时监控及原因
文章目录 0.企业高可用性标准 *** 0.1 全年无故障率(非计划内故障停机) 0.2 高可用架构方案 1. 主从复制简介 ** 2. 主从复制前提(搭建主从的过程) *** 3. 主从复制搭建(C ...