java双缓存机制_详解JVM类加载机制及类缓存问题的处理方法
前言
大家应该都知道,当一个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类加载机制及类缓存问题的处理方法相关推荐
- 详解JVM类加载机制
详解JVM类加载机制 笔者的笔记都记录在有道云里面,因为公司原因办公电脑无法使用有道云,正好借此机会整理下以前的笔记顺便当做巩固复习了,也因为记笔记的时候不会记录这些知识来源何地,所以如果发现原创后可 ...
- jvm类加载过程_详解JVM类加载
1. Java对象的创建过程 类加载检查 ===> 分配内存 ===> 初始化零值 ===> 设置对象头 ===> 执行init方法 1.1 类加载检查 虚拟机遇到一条new指 ...
- Java开发常见面试题详解(JVM)_2
Java开发常见面试题详解(JVM)_2 JVM 问题 详解 JVM垃圾回收的时候如何确定垃圾?是否知道什么是GC Roots link 你说你做过JVM调优和参数配置,请问如何盘点查看JVM系统默认 ...
- java 标量替换_详解jvm中的标量替换
概述 通常在java中创建一个对象,大家都认为是在堆中创建. 在jdk6开始有逃逸分析,标量替换等技术,关于在堆中创建对象不再绝对. 关于标量替换,通过以下几点进行概述: 逃逸分析 标量替换是什么 测 ...
- java 配置文件的路径_详解java配置文件的路径问题
详解java配置文件的路径问题 详解java配置文件的路径问题 各种语言都有自己所支持的配置文件,配置文件中有很多变量是经常改变的.不将程序中的各种变量写死,这样能更方便地脱离程序本身去修改相关变量设 ...
- java反射机制深入详解_Java反射机制深入详解
原标题:Java反射机制深入详解 一.概念 反射就是把Java的各种成分映射成相应的Java类. Class类的构造方法是private,由JVM创建. 反射是java语言的一个特性,它允程序在运行时 ...
- java新建一个女朋友_详解java创建一个女朋友类(对象啥的new一个就是)==建造者模式,一键重写...
创建一个女朋友,她有很多的属性,比如:性别,年龄,身高,体重,类型等等,虽然每个女朋友都有这些属性,但是每个人找女朋友的要求都是不一样的,有的人喜欢男的,有的人喜欢女的,有的喜欢胖的,不同的人可以根据 ...
- vue 加载页面时触发时间_详解Vue.js在页面加载时执行某个方法
详解Vue.js在页面加载时执行某个方法 jQuery中可以这样写 vue中,如果要达到相同效果,可以使用vue的生命周期函数,如create或者mounted 附上vue.js的生命周期函数执行流程 ...
- python三维图切片提取_详解Python二维数组与三维数组切片的方法
如果对象是二维数组,则切片应当是x[:]的形式,里面有一个冒号,冒号之前和之后分别表示对象的第0个维度和第1个维度: 如果对象是三维数组,则切片应当是x[::],里面有两个冒号,分割出三个间隔,三个间 ...
最新文章
- Java项目:人事管理系统(java+javaweb+jdbc)
- php 为啥报错,php Soap 报错 求大神帮忙看看为什么
- Pycharm 导入 Python 包、模块
- 【转】使IFRAME在iOS设备上支持滚动
- rsync 配置详解
- php完美导出word,PHP使用phpword生成word文档
- angularJS1.6.3个人理解(后续更新4.4.7)
- java web资源目录下_Java Web项目中的各种资源的路径写法
- Atitit java js groupby 分组操作法
- 上位机与PLC基于Fins/UDP通信
- 单车骑行红叶谷70KM
- 2022年,英文科技论文写作与学术报告期末考试答案(仅供参考)
- vue实战项目-喵喵电影 学习笔记(1)
- 是谁说的测试工资高的,应届毕业生,面试测试岗5k薪资都没人要.....
- 揭秘:恒生电子到底是干什么的
- outline如何使用呢?从秘钥获取到到文件配置完整教程分享
- 对达芬奇密码电影版的两点评价
- eclipse关不掉
- MYSQL中查询怎么判断一个字段包含英文?
- 学习方法 今天的收获
热门文章
- cmake find_package路径详解
- php怎么实现点卡充值,利用自动发卡程序的点卡充值传奇脚本
- php读取ds18b20,DS18B20_单总线协议
- redis哨兵模式没有切换主机_Redis哨兵(Sentinel)模式
- 破坏计算机信息系统功能罪,破坏计算机信息系统罪
- java 机器码 虚拟机_Java虚拟机:源码到机器码
- android sqlite存储数据,Android之SQLite数据存储
- 测试项目:车牌检测,行人检测,红绿灯检测,人流检测,目标识别
- pytorch 查看参数是否被训练 require_grad()
- linux下安装davinci