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类的加载过程 类的加载过程简单为分为三步:加载->连接->初始化 ...
随机推荐
- 进程相关API
ID与句柄 如果我们成功创建一个进程,CreateProcess函数会给我们返回一个结构体,包括四个数 据:进程编号(ID).进程句柄.线程编号(ID).线程句柄. 进程ID其实我们早见过了,通常我们 ...
- CobaltStrike4.8--云服务器搭建
系统版本选用 选择最熟悉的版本,我这边用的CentOS7.8,选用乌班图的话,会有一些命令的不一致 配置运行环境 CobaltStrike4.0支持jdk1.8的环境,4.5开始就不支持1.8了,本文 ...
- SPI总线学习笔记
SPI是串行外设接口(Serial Peripheral Interface)的缩写,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚,同时为PCB的布局上节省空 ...
- Robot Framework 自动化测试随笔(二)
二.Web自动化(1) 1.安装selenium2library库 pip install robotframework-selenium2library 2.指定报告的生成路径 在[Run]标签 ...
- 显示Label标签
1 from PyQt5.QtWidgets import QApplication, QLabel, QWidget, QVBoxLayout 2 from PyQt5.QtCore import ...
- Spring Boot 最佳实践
本文翻译自国外论坛 medium,原文地址:https://medium.com/@raviyasas/spring-boot-best-practices-for-developers-3f3bdf ...
- Go面经 | 成都Go面试这么卷?卷王介绍:游戏行业 3年经验 20k+
Go最新面经分享:算法.并发模型.缓存落盘.etcd.actor模型.epoll等等... 本文先分享2段面经,文末总结了关键问题的复盘笔记.一定要看到最后! 求职者情况 分享一下好友的最新面经. 简 ...
- API接口开发管理平台--多领域企业数字化管理解决方案
随着数字化时代的到来,企业需要进行数字化转型才能更好地适应市场需求和用户需求.而API接口则是数字化转型中的重要组成部分,可以帮助企业更好地管理信息,提高效率.本文将介绍挖数据解决方案--API接口开 ...
- YShow性能测试平台搭建
ShowSlow安装 ShowSlow是一个YSlow性能数据收集平台,用于将收集的性能数据 ShowSlow是用php实现的,所以我们需要搭建一台服务器来接收YSlow数据 我搭建的环境是:ubun ...
- from my mac
hello