前言

大家应该都知道,当一个Java项目启动的时候,JVM会找到main方法,根据对象之间的调用来对class文件和所引用的jar包中的class文件进行加载(其步骤分为加载、验证、准备、解析、初始化、使用和卸载),方法区中开辟内存来存储类的运行时数据结构(包括静态变量、静态方法、常量池、类结构等),同时在堆中生成相应的Class对象指向方法区中对应的类运行时数据结构。

用最简单的一句话来概括,类加载的过程就是JVM根据所需的class文件的路径,通过IO流的方式来读取class字节码文件,并通过一系列解析初始化等步骤来注入到内存。 java中的类加载器有:BootstrapClassLoader(最上层)、ExtClassLoader、AppClassLoader、以及用户自定义的ClassLoader(最下层)。JVM对于不同种类的jar包(或class文件),会有不同种类的类加载器进行加载。

对应关系如下:

BootstrapClassLoader  用于加载JVM运行所需要的类:

JAVA_HOME/jre/lib/resources.jar:

JAVA_HOME/jre/lib/rt.jar:

JAVA_HOME/jre/lib/sunrsasign.jar:

JAVA_HOME/jre/lib/jsse.jar:

JAVA_HOME/jre/lib/jce.jar:

JAVA_HOME/jre/lib/charsets.jar:

JAVA_HOME/jre/lib/jfr.jar:

JAVA_HOME/jre/classes

ExtClassLoader 用于加载扩展类:

../Java/Extensions:

../JAVA_HOME/jre/lib/ext:    ../Library/Java/Extensions:/Network/Library/Java/Extensions:

../System/Library/Java/Extensions:

../lib/java

AppClassLoader 用于加载我们项目中ClassPath下所创建的类和jar包中引用的类。

整个类加载,是通过一种叫做双亲委派的机制来进行加载。

举例来说,一个类被最下层的加载器(用户自定义ClassLoader)进行加载,此加载器首先会调用上一层的加载器(AppClassLoader)进行加载,而AppClassLoader会继续转交给上层(ExtClassLoader)的加载器进行加载,直到BootstrapClassLoader。  如果BootstrapClassLoader所加载的类路径找不到此类,那么才会交给下一层的加载器(ExtClassLoader)进行加载,如果找不到此类,继续交给下一层(AppClassLoader)进行加载。以此类推,如果用户自定义的ClassLoader也找不到此类,那么程序就会抛出一个ClassNotFoundError。

整个加载过程图示如下:

(图片引用自:https://www.cnblogs.com/xing901022/p/4574961.html)

类加载源的源码跟踪如下(在此对源码进行了适当的简化),读者可以点入源码进行查看:

package java.lang.ClassLoader;

import ....

protected Class> loadClass(String name, boolean resolve)

throws ClassNotFoundException

{

synchronized (getClassLoadingLock(name)) {

// First,在虚拟机内存中查找是否已经加载过此类...类缓存的主要问题所在!!!

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) {

//调用此类加载器所实现的findClass方法进行加载

c = findClass(name);

}

}

if (resolve) {

resolveClass(c);

}

return c;

}

}

在源码中可以完全领略到双亲委派机制的过程,其中最重要的三句代码已经进行了标注:

findLoadedClass(在虚拟机内存中查找是否已经加载过此类...类缓存的主要问题所在!!!)

parent.loadClass(先让上一层加载器进行加载)

findClass(调用此类加载器所实现的findClass方法进行加载)

如果用户需要自定义加载器,加载自己指定路径的class文件,需要继承ClassLoader,并实现findClass(String name)方法。举例如下:

package com.linuxidc.utils;

import java.io.ByteArrayOutputStream;

import java.io.FileInputStream;

import java.io.IOException;

import java.io.InputStream;

public class ServiceClassLoader extends ClassLoader{

private String classPath;

public ServiceClassLoader(String classPath) {

this.classPath = classPath;

}

/**

* 重写父类的findClass 方法。 父类的loadClass会调用此方法

*/

@Override

protected Class> findClass(String name) throws ClassNotFoundException {

Class> c = null;

byte[] classData = getClassData(name);

if (classData!=null) {

c = defineClass(name, classData, 0, classData.length);

}else {

throw new ClassNotFoundException();

}

return c;

}

// 将class文件通过IO流读取,转化为字节数组

private byte[] getClassData(String name) {

String path = classPath + "/"+ name.replace('.', '/') + ".class";

InputStream iStream = null;

ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();

try {

iStream = new FileInputStream(path);

byte[] buffer = new byte[1024];

int temp = 0;

while ((temp = iStream.read(buffer))!=-1) {

byteArrayOutputStream.write(buffer, 0, temp);

}

if (byteArrayOutputStream!=null) {

return byteArrayOutputStream.toByteArray();

}

} catch (Exception e) {

e.printStackTrace();

}finally {

try {

if (iStream!=null) {

iStream.close();

}

} catch (IOException e) {

e.printStackTrace();

}

try {

if (byteArrayOutputStream!=null) {

byteArrayOutputStream.close();

}

} catch (IOException e) {

e.printStackTrace();

}

}

return null;

}

}

对类加载器的使用代码如下:

ServiceClassLoader serviceClassLoader = new ServiceClassLoader("c:\myclass");

Czlass> c = ServiceClassLoader.loadClass("com.linuxidc.service.Myclass");

如果 用同一个 ServiceClassLoader 对象去加载同一个Class文件多次,每次加载后的Class对象为同一个! 然而如果new不同的自定义ClassLoader去加载同一个Class文件,则每次会返回不同的Class对象。

注意: 不能将所要加载的Class文件放到classpath目录及其任何子目录下,否则会被AppClassLoader优先加载 (这是由于类加载采用双亲委派机制,同时AppClassLoader可以加载所有在classpath下的class文件), 每次都是同一个AppClassLoader进行加载,因此会出现类缓存问题。

这样就解决了通常在JVM类加载时,直接使用反射出现的类缓存的问题。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。

java双缓存机制_详解JVM类加载机制及类缓存问题的处理方法相关推荐

  1. 详解JVM类加载机制

    详解JVM类加载机制 笔者的笔记都记录在有道云里面,因为公司原因办公电脑无法使用有道云,正好借此机会整理下以前的笔记顺便当做巩固复习了,也因为记笔记的时候不会记录这些知识来源何地,所以如果发现原创后可 ...

  2. jvm类加载过程_详解JVM类加载

    1. Java对象的创建过程 类加载检查 ===> 分配内存 ===> 初始化零值 ===> 设置对象头 ===> 执行init方法 1.1 类加载检查 虚拟机遇到一条new指 ...

  3. Java开发常见面试题详解(JVM)_2

    Java开发常见面试题详解(JVM)_2 JVM 问题 详解 JVM垃圾回收的时候如何确定垃圾?是否知道什么是GC Roots link 你说你做过JVM调优和参数配置,请问如何盘点查看JVM系统默认 ...

  4. java 标量替换_详解jvm中的标量替换

    概述 通常在java中创建一个对象,大家都认为是在堆中创建. 在jdk6开始有逃逸分析,标量替换等技术,关于在堆中创建对象不再绝对. 关于标量替换,通过以下几点进行概述: 逃逸分析 标量替换是什么 测 ...

  5. java 配置文件的路径_详解java配置文件的路径问题

    详解java配置文件的路径问题 详解java配置文件的路径问题 各种语言都有自己所支持的配置文件,配置文件中有很多变量是经常改变的.不将程序中的各种变量写死,这样能更方便地脱离程序本身去修改相关变量设 ...

  6. java反射机制深入详解_Java反射机制深入详解

    原标题:Java反射机制深入详解 一.概念 反射就是把Java的各种成分映射成相应的Java类. Class类的构造方法是private,由JVM创建. 反射是java语言的一个特性,它允程序在运行时 ...

  7. java新建一个女朋友_详解java创建一个女朋友类(对象啥的new一个就是)==建造者模式,一键重写...

    创建一个女朋友,她有很多的属性,比如:性别,年龄,身高,体重,类型等等,虽然每个女朋友都有这些属性,但是每个人找女朋友的要求都是不一样的,有的人喜欢男的,有的人喜欢女的,有的喜欢胖的,不同的人可以根据 ...

  8. vue 加载页面时触发时间_详解Vue.js在页面加载时执行某个方法

    详解Vue.js在页面加载时执行某个方法 jQuery中可以这样写 vue中,如果要达到相同效果,可以使用vue的生命周期函数,如create或者mounted 附上vue.js的生命周期函数执行流程 ...

  9. python三维图切片提取_详解Python二维数组与三维数组切片的方法

    如果对象是二维数组,则切片应当是x[:]的形式,里面有一个冒号,冒号之前和之后分别表示对象的第0个维度和第1个维度: 如果对象是三维数组,则切片应当是x[::],里面有两个冒号,分割出三个间隔,三个间 ...

最新文章

  1. Java项目:人事管理系统(java+javaweb+jdbc)
  2. php 为啥报错,php Soap 报错 求大神帮忙看看为什么
  3. Pycharm 导入 Python 包、模块
  4. 【转】使IFRAME在iOS设备上支持滚动
  5. rsync 配置详解
  6. php完美导出word,PHP使用phpword生成word文档
  7. angularJS1.6.3个人理解(后续更新4.4.7)
  8. java web资源目录下_Java Web项目中的各种资源的路径写法
  9. Atitit java js groupby 分组操作法
  10. 上位机与PLC基于Fins/UDP通信
  11. 单车骑行红叶谷70KM
  12. 2022年,英文科技论文写作与学术报告期末考试答案(仅供参考)
  13. vue实战项目-喵喵电影 学习笔记(1)
  14. 是谁说的测试工资高的,应届毕业生,面试测试岗5k薪资都没人要.....
  15. 揭秘:恒生电子到底是干什么的
  16. outline如何使用呢?从秘钥获取到到文件配置完整教程分享
  17. 对达芬奇密码电影版的两点评价
  18. eclipse关不掉
  19. MYSQL中查询怎么判断一个字段包含英文?
  20. 学习方法 今天的收获

热门文章

  1. cmake find_package路径详解
  2. php怎么实现点卡充值,利用自动发卡程序的点卡充值传奇脚本
  3. php读取ds18b20,DS18B20_单总线协议
  4. redis哨兵模式没有切换主机_Redis哨兵(Sentinel)模式
  5. 破坏计算机信息系统功能罪,破坏计算机信息系统罪
  6. java 机器码 虚拟机_Java虚拟机:源码到机器码
  7. android sqlite存储数据,Android之SQLite数据存储
  8. 测试项目:车牌检测,行人检测,红绿灯检测,人流检测,目标识别
  9. pytorch 查看参数是否被训练 require_grad()
  10. linux下安装davinci