java jdk 类加载机制_JDK源码阅读之类加载

java类加载

类的生命周期(类加载过程)

LLIUU+VPR

加载(Loading)

链接(Linking)

验证(Verification)

准备(Preparation)

解析(Resolution)

初始化(Initialization)

使用(Using)

卸载(Unloading) 类

类加载器种类

BootstrapClassLoader:C++编写,负责加载java核心类库

Launcher.ExtClassLoader:Launcher中的内部类,parent == null

Launcher.AppClassLoader:Launcher中的内部类,parent == ExtClassLoader

用户自定义ClassLoader:继承自ClassLoader,parent == AppClassLoader

e1f12135292e1c4288fc34c7f8c2aa6a.png

类加载机制

java中默认的类加载机制是双亲委派模式。

ClassLoader中关键的方法说明:

loadClass // 类加载入口,包含下面这些步骤

=> findLoadedClass => findLoadedClass0 // 先从缓存中查询一下,看看目标类是否已加载过

=> findBootstrapClassOrNull => findBootstrapClass // 用Bootstrap类加载器进行加载

=> findClass // 读取字节码文件,然后加载字节码文件

=> defineClass // 加载字节码文件

=> preDefineClass // 加载前的检查

=> defineClassSourceLocation // 定义类加载的路径

=> defineClass1/defineClass2 // 调用native方法加载类

=> postDefineClass //

=> resolveClass => resolveClass0

ClassLoader 部分源码:

package java.lang;

import java.io.InputStream;

...

public abstract class ClassLoader {

private final ClassLoader parent;

// -- Class --

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;

}

}

// findClass由子类去实现

protected Class> findClass(String name) throws ClassNotFoundException {

throw new ClassNotFoundException(name);

}

// defineClass 加载类

protected final Class> defineClass(String name, byte[] b, int off, int len,

ProtectionDomain protectionDomain)

throws ClassFormatError

{

protectionDomain = preDefineClass(name, protectionDomain);

String source = defineClassSourceLocation(protectionDomain);

Class> c = defineClass1(name, b, off, len, protectionDomain, source);

postDefineClass(c, protectionDomain);

return c;

}

private ProtectionDomain preDefineClass(String name,

ProtectionDomain pd)

{

if (!checkName(name))

throw new NoClassDefFoundError("IllegalName: " + name);

// Note: Checking logic in java.lang.invoke.MemberName.checkForTypeAlias

// relies on the fact that spoofing is impossible if a class has a name

// of the form "java.*"

if ((name != null) && name.startsWith("java.")) {

throw new SecurityException

("Prohibited package name: " +

name.substring(0, name.lastIndexOf('.')));

}

if (pd == null) {

pd = defaultDomain;

}

if (name != null) checkCerts(name, pd.getCodeSource());

return pd;

}

private String defineClassSourceLocation(ProtectionDomain pd)

{

CodeSource cs = pd.getCodeSource();

String source = null;

if (cs != null && cs.getLocation() != null) {

source = cs.getLocation().toString();

}

return source;

}

private void postDefineClass(Class> c, ProtectionDomain pd)

{

if (pd.getCodeSource() != null) {

Certificate certs[] = pd.getCodeSource().getCertificates();

if (certs != null)

setSigners(c, certs);

}

}

private native Class> defineClass0(String name, byte[] b, int off, int len,

ProtectionDomain pd);

private native Class> defineClass1(String name, byte[] b, int off, int len,

ProtectionDomain pd, String source);

private native Class> defineClass2(String name, java.nio.ByteBuffer b,

int off, int len, ProtectionDomain pd,

String source);

protected final void resolveClass(Class> c) {

resolveClass0(c);

}

private native void resolveClass0(Class> c);

private Class> findBootstrapClassOrNull(String name) {

if (!checkName(name)) return null;

return findBootstrapClass(name);

}

// return null if not found

private native Class> findBootstrapClass(String name);

protected final Class> findLoadedClass(String name) {

if (!checkName(name))

return null;

return findLoadedClass0(name);

}

private native final Class> findLoadedClass0(String name);

// -- Resource --

...

// -- Hierarchy --

...

// -- Package --

...

// -- Native library access --

...

// -- Assertion management --

...

}

双亲委派

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/269642.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2017.4.11 AM

练恋有词U28单元单词及应用 转载于:https://www.cnblogs.com/bgd140201219/p/6696472.html

volatile不具备原子性

1、理解原子性&#xff1a; 上面说到volatile不具备原子性&#xff0c;那么原子性到底是什么呢&#xff1f;先看如下代码 public class TestVolatile {public static void main(String[] args) {AtomicDemo atomicDemo new AtomicDemo();for (int x 0; x < 10; x) {new Th…

CountDownLatch的理解和使用

闭锁 CountDownLatch概念 CountDownLatch是一个同步工具类&#xff0c;用来协调多个线程之间的同步&#xff0c;或者说起到线程之间的通信&#xff08;而不是用作互斥的作用&#xff09;。 CountDownLatch能够使一个线程在等待另外一些线程完成各自工作之后&#xff0c;再继…

汇编语言学习笔记(五)

十六、数组 数组的基本表示方法 numary sdword 2,5,7 numary数组中有三个元素&#xff0c;为sdword类型&#xff0c;分别为2,5,7 empary sdword ?, ?,? empary数组为sdword类型元素&#xff0c;未初始化。 如果数组元素很多可通过 zeroary sdword 100 dup(0) zeroary数组中有…

[linux命令技巧] mkdir -p

mkdir {dirname} 只能建立单个目录。mkdir的-p选项允许你一次性创建多层次的目录&#xff0c;而不是一次只创建单独的目录。例如&#xff0c;我们要在当前目录创建目录/home/a/b (/home为空)&#xff0c;使用命令cd /home mkdir a cd a mkdir b 当然可以&#xff0c;但是使用 m…

Git文件四种状态

git status可以看到文件是untracked 未跟踪状态 git add . 把文件添加到暂存区 接着 git status可以看到是staged状态 git commit -m "备注"

java中io流是类吗_Java中的IO流

今天刚刚看完java的io流操作&#xff0c;把主要的脉络看了一遍&#xff0c;不能保证以后使用时都能得心应手&#xff0c;但是最起码用到时知道有这么一个功能可以实现&#xff0c;下面对学习进行一下简单的总结&#xff1a;IO流主要用于硬板、内存、键盘等处理设备上得数据操作…

动态规划(冬令营课堂笔记)

简单问题 01背包 012背包 部分背包 机器分配 烽火传递 花店橱窗问题 简单问题 01背包 一个容量为m的背包&#xff0c;有n个物品&#xff0c;第i个物品的体积为wi&#xff0c;价值为ci。选择若干物品&#xff0c;使得体积总和不超过m的情况下价值总和最大。 n<100&#xff0c…

Git使用命令行回退版本git reset --hard

git log--oneline --oneline 标记的作用是把每一个提交信息压缩为一行。默认情况下只会展示提交 ID与提交信息的首行。git log --oneline的结果如下 方法一&#xff1a; git reset --hard~回退几个版本 git reset --hard~3 表示回退三个版本&#xff0c;即从8309203回到93b1…

Git分支命令学习使用

git branch 查看分支&#xff0c;如下图 黄色的有*号的表示当前分支 git branch 分支名 表示创建一个新分支 git checkout 分支名 表示切换到这个分支 git checkout -b 分支名 表示创建这个新分支并且切换到这个分支上

php提交表单处理,PHP表单处理

我们可以在PHP中创建和使用表单。要获取表单数据&#xff0c;需要使用PHP超级元组&#xff1a;$_GET和$_POST。表单请求可以是get或post。 要从get请求中检索数据&#xff0c;需要使用$_GET&#xff0c;而$_POST用于检索post请求中的数据。PHP GET表单GET请求是表单的默认请求。…

git中--soft和--mixed和--hard区别

想cvbnm&#xff0c;。、、、、、

怎样在mac系统里将文件拷贝到移动硬盘教程

一&#xff1a;下载这个mounty软件 地址https://mounty.app/ 下载安装 打开后菜单栏显示一座山&#xff0c;如下 接着就可以移动了

Error:java: 无效的目标发行版: 11解决方案

我们在使用Idea开发中如果遇到你的JDK版本有“无效的目标发行版: 11”冲突时&#xff0c;要修改以下JDK版本

java证明ArrayList是线程不安全的

证明ArrayList是线程不安全的 我们开启100个线程.每个线程向List加100个数据,那么当所有线程执行完成之后应该是10000条,然后就对比一下结果,看看是否为10000条. thread.join(); 是让主线程等待所有的子线程执行完,才执行接下来的语句 运行结果为9988,而且每次运行结果还不一…

[CSS] 点击事件触发的动画

源码 https://github.com/YouXianMing/CSS-Animations/tree/master/Event 效果 细节 1) 一个完整的可回溯的动画至少包括了两种状态,以及两种状态的动画设置,还有其关键帧设置. 2) 设置的值其实只有A,B两种状态而已 3) 动画开始的时候,只设置一个状态值,而不设置状态的动画设置…

discuz邮件设置PHP,Discuz!6.0—如何配置发送邮件的参数

邮件发送方式个教程教大家如何配置邮件参数&#xff0c;主要介绍以下两个个方面&#xff1a;邮件发送方式邮件头的分隔符一、邮件发送方式&#xff1a;Discuz!6.0支持如下三种邮件发送方式&#xff1a;1、通过 PHP 函数的 sendmail 发送(推荐此方式)图1介绍&#xff1a;这种方式…

在AndroidStudio中数据存储第三方数据管理Bmob的使用

---恢复内容开始--- 在日常写代码的过程中我们比较痛苦的就是数据库的建立和使用&#xff0c;那么今天来介绍一下一个第三方的数据管理平台Bmonb。 一、我们首先进入Bmob的官网创建一个账号 Bome官网网址&#xff1a;http://www.bmob.cn/ 二、打开Androidstudio新建一个项目然后…