JVM 类加载器ClassLoader源码学习笔记
类加载
在Java代码中,类型的加载、连接与初始化过程都是在程序运行期间完成的。
类型可以是Class,Interface, 枚举等。
Java虚拟机与程序的生命周期
在如下几种情况下,Java虚拟机将结束生命周期
1)执行了System.exit() 方法
2)程序正常执行结束
3)程序在执行过程中遇到了异常或者错误而异常终止。
4) 由于操作系统出现错误导致Java虚拟机进程终止。
1.JVM运行流程,JVM基本结构
2、类加载器双亲委派模型
3、ClassLoader源码解析
4、从源码分析实现自定义类加载器
一、JVM运行流程,JVM基本结构
JVM基本结构
类加载器,运行时数据区,执行引擎,本地接口
Class Files -> ClassLoader -> 运行时数据区 -> 执行引擎 -> 本地方法库
类的装载:
加载,连接(验证,准备,解析),初始化,使用,卸载
初始化: 执行类的构造器<clinit>,为类的静态变量赋初始值
构造器:
1、static变量
2、staitc{} 语句
构造方法: 实例化的对象
二、类加载器双亲委派模型
1、为什么使用双亲委派模型
避免重复加载
2、JDK已有的类加载器
BootStrap ClassLoader JVM自己的类加载器,启动加载器。(C++) 主要加载rt.jar
Extension ClassLoader extends ClassLoader 扩展类加载器 加载%Java_home%lib/ext/*.jar
APP ClassLoader extends ClassLoader 应用加载器 -> Classpath
打印Class Loader
打印parent ClassLoader
null为启动类加载器。
三、ClassLoader源码解析
ClassLoader所在的路径
1、创建关于ClassLoader的Demo
public class MyTest15 {
public static void main(String[] args) {
String[] strings = new String[2];
System.out.println(strings.getClass());
System.out.println(strings.getClass().getClassLoader()); //启动类加载器
System.out.println("--------------------"); MyTest15[] myTest15s = new MyTest15[2];
System.out.println(myTest15s.getClass().getClassLoader()); //AppClassLoader System.out.println("--------------------"); int[] ints = new int[2];
System.out.println(ints.getClass().getClassLoader()); //原生类型没有classLoader }
}
打印结果:
class [Ljava.lang.String;
null
--------------------
sun.misc.Launcher$AppClassLoader@18b4aac2
--------------------
null
2、定位到loadClass方法
loadClass(name,false)方法
protected Class<?> loadClass(String name, boolean resolve)
throws ClassNotFoundException
{
synchronized (getClassLoadingLock(name)) {
// First, check if the class has already been loaded
Class c = findLoadedClass(name);
if (c == null) {
long t0 = System.nanoTime();
try {
if (parent != null) {
c = parent.loadClass(name, false);
} else {
c = findBootstrapClassOrNull(name);
}
} catch (ClassNotFoundException e) {
// ClassNotFoundException thrown if class not found
// from the non-null parent class loader
} if (c == null) {
// If still not found, then invoke findClass in order
// to find the class.
long t1 = System.nanoTime();
c = findClass(name); // this is the defining class loader; record the stats
sun.misc.PerfCounter.getParentDelegationTime().addTime(t1 - t0);
sun.misc.PerfCounter.getFindClassTime().addElapsedTimeFrom(t1);
sun.misc.PerfCounter.getFindClasses().increment();
}
}
if (resolve) {
resolveClass(c);
}
return c;
}
}
3、双亲委派模型
如果parent不为空,则调用parent的loadClass方法。
findClass类的目的是自定义的ClassLoader
四、自定义类加载器 extends ClassLoader -> 完成自定义加载路径
1) 创建Demo.java文件,路径为D:/tmp/Demo.java
然后编译成class文件
javac Demo.java
2) 创建自定义类加载器
package com.classloader; import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.InputStream; public class MyClassLoader extends ClassLoader { private String path; //加载的路径
private String name; //类加载器名称 public MyClassLoader(String name, String path){
super(); //让系统类加载器成为该类的父加载器
this.name = name;
this.path = path;
} public MyClassLoader(ClassLoader parent, String name, String path){
super(parent); //显示指定父加载器
this.name = name;
this.path = path;
} /**
* 加载自定义的类,通过自定义的ClassLoader
*/
@Override
protected Class<?> findClass(String name) throws ClassNotFoundException {
byte[] data = readClassFileToByteArray(name);
return this.defineClass(name, data,0, data.length);
} /**
* 获取.class文件的字节数组
* com.classLoader.Demo ->
* D:/temp/com/classLoader/Demo.class
* @return
*/
private byte[] readClassFileToByteArray(String name) {
InputStream is = null;
byte[] returnData = null; name = name.replaceAll("\\.","/");
String filePath = this.path + name + ".class";
File file = new File(filePath); ByteArrayOutputStream os = new ByteArrayOutputStream();
try{
is = new FileInputStream(file);
int tmp = 0;
while ((tmp = is.read()) != -1){
os.write(tmp);
}
returnData = os.toByteArray();
}catch (Exception e){
e.printStackTrace();
}finally {
try {
if(is != null){
is.close();
} if(os != null){
os.close();
} }catch (Exception e2){ }
} return returnData; } @Override
public String toString() {
return this.name;
}
}
3、使用自定义类加载器
public class TestDemo { public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
MyClassLoader loader = new MyClassLoader("MyClassLoadName1", "D:/tmp/");
Class<?> c = loader.loadClass("Demo");
c.newInstance();
}
}
运行结果
Demo, MyClassLoadName1
4、测试父加载器
1)、工程里的Demo.java增加测试方法
2)、修改D盘下的Demo.java
增加包名
package com.classloader; public class Demo { public Demo(){
System.out.println("Demo, " + this.getClass().getClassLoader());
} }
路径为,并且重新生成class文件
3) 测试文件
public class TestDemo { public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException {
MyClassLoader loader = new MyClassLoader("MyClassLoadName1", "D:/tmp/");
Class<?> c = loader.loadClass("com.classloader.Demo");
c.newInstance();
}
}
运行结果:
说明使用的是父类加载器,加载的是工程里的那个Demo.class文件。
修改如下:
传入null,说明父加载器为启动加载器。
显示结果:
Demo, MyClassLoadNameChild
JVM 类加载器ClassLoader源码学习笔记的更多相关文章
- 第五章 类加载器ClassLoader源码解析
说明:了解ClassLoader前,先了解 第四章 类加载机制 1.ClassLoader作用 类加载流程的"加载"阶段是由类加载器完成的. 2.类加载器结构 结构:Bootstr ...
- 类加载器ClassLoader源码解析
1.ClassLoader作用 类加载流程的"加载"阶段是由类加载器完成的. 2.类加载器结构 结构:BootstrapClassLoader(祖父)-->ExtClassL ...
- JUC源码学习笔记4——原子类,CAS,Volatile内存屏障,缓存伪共享与UnSafe相关方法
JUC源码学习笔记4--原子类,CAS,Volatile内存屏障,缓存伪共享与UnSafe相关方法 volatile的原理和内存屏障参考<Java并发编程的艺术> 原子类源码基于JDK8 ...
- Spring 源码学习笔记10——Spring AOP
Spring 源码学习笔记10--Spring AOP 参考书籍<Spring技术内幕>Spring AOP的实现章节 书有点老,但是里面一些概念还是总结比较到位 源码基于Spring-a ...
- Qt Creator 源码学习笔记04,多插件实现原理分析
阅读本文大概需要 8 分钟 插件听上去很高大上,实际上就是一个个动态库,动态库在不同平台下后缀名不一样,比如在 Windows下以.dll结尾,Linux 下以.so结尾 开发插件其实就是开发一个动态 ...
- async-validator 源码学习笔记(五):Schema
系列文章: 1.async-validator 源码学习(一):文档翻译 2.async-validator 源码学习笔记(二):目录结构 3.async-validator 源码学习笔记(三):ru ...
- Spring 源码学习笔记11——Spring事务
Spring 源码学习笔记11--Spring事务 Spring事务是基于Spring Aop的扩展 AOP的知识参见<Spring 源码学习笔记10--Spring AOP> 图片参考了 ...
- Spring源码学习笔记12——总结篇,IOC,Bean的生命周期,三大扩展点
Spring源码学习笔记12--总结篇,IOC,Bean的生命周期,三大扩展点 参考了Spring 官网文档 https://docs.spring.io/spring-framework/docs/ ...
- Underscore.js 源码学习笔记(下)
上接 Underscore.js 源码学习笔记(上) === 756 行开始 函数部分. var executeBound = function(sourceFunc, boundFunc, cont ...
随机推荐
- Linux命令——getconf
转自:灵活使用getconf命令来获取系统信息 简介 getconf本身是个ELF可执行文件,用于获取系统信息 用法 getconf -a可以获取全部系统信息 对于这个命令,记住几个常用的信息获取方法 ...
- SQL进阶系列之0窗口函数
窗口函数 What's 窗口函数? 窗口函数也称为OLAP(OnLine Analytical Processing)函数,目前MySQL还不支持. 窗口函数的语法 <窗口函数> OVER ...
- SUSE 12安装详解
1.部署步骤 1.1.启动安装程序 在启动页面上选择Installation,然后按Enter键,这将载入SUSE Linux服务器安装程序并以普通模式安装. 1.2.选择安装语言 Language和 ...
- Spring4- 01 - Spring框架简介及官方压缩包目录介绍- Spring IoC 的概念 - Spring hello world环境搭建
一. Spring 框架简介及官方压缩包目录介绍 主要发明者:Rod Johnson 轮子理论推崇者: 2.1 轮子理论:不用重复发明轮子. 2.2 IT 行业:直接使用写好的代码. Spring 框 ...
- vue 标签页以及标签页赋值
背景: 使用vue增加了标签页,点击不同标签页传给后端的值不一样,用来做区分,如图: vue代码如下: 使用 form.PageA form.PageB ,后端接收到的值 first.second ...
- Oracle 序号函数
Oracle提供的序号函数:以emp表为例:1: rownum 最简单的序号 但是在order by之前就确定值.select rownum,t.* from emp t order by ename ...
- DevExpress21:SplashScreenManager控件实现启动闪屏和等待信息窗口
DevExpress中SplashScreenManager这个控件的主要作用就是显示程序集加载之前的进度条显示和进行耗时操作时候的等待界面. 一.SplashScreenManager控件的使用 1 ...
- PHP——汉字完美转为ASCII码
前言 对接联通的接口,让我学会了不少PHP偏门函数....,主要对方用的py,我这用的PHP,人家一个函数解决了, 我这还要自己写方法,也是比较蛋疼,但是学到东西还是很开心的~ 代码 字符串转为ASC ...
- C++类分号(;)问题
环境:vs2010 问题:今天编代码过程中发现好多很奇怪的错误,我以为昨天调了下编译器才出问题了.搞了好久,代码注释掉很多还是不行,并且错误还一直在变化.问题大概如下: (照片上传不了) .error ...
- VS调试web api服务
vs2013开发web api service时,使用vs开发服务器调试没有问题,但将项目放到另一台电脑调试(vs2010),总会提示 无法再以下端口启动asp.net开发服务器 错误:通常每个套接字 ...