类加载

在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源码学习笔记的更多相关文章

  1. 第五章 类加载器ClassLoader源码解析

    说明:了解ClassLoader前,先了解 第四章 类加载机制 1.ClassLoader作用 类加载流程的"加载"阶段是由类加载器完成的. 2.类加载器结构 结构:Bootstr ...

  2. 类加载器ClassLoader源码解析

    1.ClassLoader作用 类加载流程的"加载"阶段是由类加载器完成的. 2.类加载器结构 结构:BootstrapClassLoader(祖父)-->ExtClassL ...

  3. JUC源码学习笔记4——原子类,CAS,Volatile内存屏障,缓存伪共享与UnSafe相关方法

    JUC源码学习笔记4--原子类,CAS,Volatile内存屏障,缓存伪共享与UnSafe相关方法 volatile的原理和内存屏障参考<Java并发编程的艺术> 原子类源码基于JDK8 ...

  4. Spring 源码学习笔记10——Spring AOP

    Spring 源码学习笔记10--Spring AOP 参考书籍<Spring技术内幕>Spring AOP的实现章节 书有点老,但是里面一些概念还是总结比较到位 源码基于Spring-a ...

  5. Qt Creator 源码学习笔记04,多插件实现原理分析

    阅读本文大概需要 8 分钟 插件听上去很高大上,实际上就是一个个动态库,动态库在不同平台下后缀名不一样,比如在 Windows下以.dll结尾,Linux 下以.so结尾 开发插件其实就是开发一个动态 ...

  6. async-validator 源码学习笔记(五):Schema

    系列文章: 1.async-validator 源码学习(一):文档翻译 2.async-validator 源码学习笔记(二):目录结构 3.async-validator 源码学习笔记(三):ru ...

  7. Spring 源码学习笔记11——Spring事务

    Spring 源码学习笔记11--Spring事务 Spring事务是基于Spring Aop的扩展 AOP的知识参见<Spring 源码学习笔记10--Spring AOP> 图片参考了 ...

  8. Spring源码学习笔记12——总结篇,IOC,Bean的生命周期,三大扩展点

    Spring源码学习笔记12--总结篇,IOC,Bean的生命周期,三大扩展点 参考了Spring 官网文档 https://docs.spring.io/spring-framework/docs/ ...

  9. Underscore.js 源码学习笔记(下)

    上接 Underscore.js 源码学习笔记(上) === 756 行开始 函数部分. var executeBound = function(sourceFunc, boundFunc, cont ...

随机推荐

  1. mysql学习之基础篇01

    大概在一周前看了燕十八老师讲解的mysql数据库视频,也跟着学了一周,我就想把我这一周所学的知识跟大家分享一下:因为是第一次写博客,所以可能会写的很烂,请大家多多包涵.文章中有不对的地方还请大家指出来 ...

  2. 基于k8s集群部署prometheus监控etcd

    目录 基于k8s集群部署prometheus监控etcd 1.背景和环境概述 2.修改prometheus配置 3.检查是否生效 4.配置grafana图形 基于k8s集群部署prometheus监控 ...

  3. gdb调试(二)

    继续研究gdb相关的调试技巧,话不多说进入正题: 查看运行时数据: 这个上节中已经用过了,这里就不多说了,比较简单 还是有上节中的simple.c例子,不过得稍微做一些修改为了使用这些命令: simp ...

  4. 微信小程序API~用户信息

    UserInfo 用户信息 属性 string nickName 用户昵称 string avatarUrl 用户头像图片的 URL.URL 最后一个数值代表正方形头像大小(有 0.46.64.96. ...

  5. CF1167E. Range Deleting

    题意 给定长度为\(n\)的数组\(a\),其中任意\(a_i \leq x\) 定义\(f(l,r)\)为删除\(a\)中值域在\([l,r]\)的数后剩余的数组. 统计满足\(1\leq l \l ...

  6. 《你们都是魔鬼吗》实验十二 团队作业八:Alpha冲刺

    <你们都是魔鬼吗>第八次团队作业:Alpha冲刺 项目 内容 这个作业属于哪个课程 任课教师博客主页链接 这个作业的要求在哪里 作业链接地址 团队名称 你们都是魔鬼吗 作业学习目标 软件测 ...

  7. (一)WCF基础

    我们近期在做项目的时候用到了WCF,之前已经看了部分视频,对于WCF有了一定的了解,但仅限于能够根据搭建好的框架使用WCF,还不了解.所以就进行了研究,这样既有实践也能增加理论,二者结合,使用起来更胜 ...

  8. vscode beautiful配置

    在工作目录下建立.jsbeautifyrc文件 官方文档 { "brace_style": "none,preserve-inline", "inde ...

  9. C# 6.0 中的新增功能(.NET Framework 4.6 与 Visual Studio 2015 )

    C#6.0 在 2015 年7月随着.NET Framework 4.6 一同发布,后期发布了.NET Framework 4.6.1,4.6.2. 一.自动属性初始化(Auto-property i ...

  10. Nutch2.1+solr3.6.1+mysql5.6问题

    1.Nutch2.1问题 1.1 问题:导入完成后,Nutch2.1里面runtime仍旧不能运行,出现jobfailed等错误. 解决:runtime里的nutch调试过程和导入Eclipse差不多 ...