1、类加载器(ClassLoader)负责加载class文件,class文件在文件开头有特定的文件标识,并且ClassLoader只负责 class 文件的加载,至于class文件是否能够运行则由Execution Engine决定;

  

2、类的加载过程

加载class文件 ---> 连接(验证、准备、解析class文件)----> 初始化类的对象 ----> 使用类的对象 -----> 卸载。

(1)加载class文件

  取类的二进制字节流,通过类的全量命名,将静态存储结构房到方法区中,Class(类的定义或结构,即类对象)放到堆中;

(2)初始化类对象

  执行类的构造器<clinit>,为类的静态变量赋予正确的初值;

  构造器包含:static变量、static块;根据代码中的顺序来决定先执行哪个;

  先执行构造器(static变量、static块),再执行构造方法;

3、JDK中的类加载器:

  <1> 启动类加载器(BootStrap ClassLoader): C++编写的,虚拟机自带的加载器;加载$JAVA_HOME/jre/lib/rt.jar 

  <2> 扩展类加载器(Extension ClassLoader):java编写;加载$JAVA_HOME/jre/lib/ext/*.jar

  <3> 应用程序类加载器(App ClassLoader):java编写;也叫系统类加载器,加载当前应用的 $classpath 下所有类

    <4> 用户自定义加载器:Java.Lang.ClassLoader 的子类,用户可以定制类的加载方式;

  

3、类加载器说明

 第一步:将 MyHello 类打包成 jar包,然后放入$JAVA_HOME/jre/lib/ext/ 目录下。  MyHello内容如下:

public class MyHello
{
public static void main(String[] args)
{ }
}

 第二步:编写如下代码

public class Demo1
{
public static void main(String[] args) throws ClassNotFoundException {
//启动类加载器(BootStrap ClassLoader)
Object object = new Object();
System.out.println("Object-classLoader: " + object.getClass().getClassLoader());
//结果: Object-classLoader: null //扩展类加载器 (Extension ClassLoader)
Class myHello = Class.forName("com.yufeng.jvm.hello.MyHello");
System.out.println("MyHello-classLoader: " + myHello.getClassLoader());
//结果:MyHello-classLoader: sun.misc.Launcher$ExtClassLoader@7ea987ac //应用类加载器 (App ClassLoader)
Demo1 classLoaderDemo1 = new Demo1();
System.out.println("Demo1-classLoader: " + classLoaderDemo1.getClass().getClassLoader());
//①结果:Demo1-classLoader: sun.misc.Launcher$AppClassLoader@18b4aac2 System.out.println("Demo1-classLoader-parent: " + classLoaderDemo1.getClass().getClassLoader().getParent());
//②结果:Demo1-classLoader-parent: sun.misc.Launcher$ExtClassLoader@2503dbd3 System.out.println("Demo1-classLoader-parent-parent: " + classLoaderDemo1.getClass().getClassLoader().getParent().getParent());
//③结果:Demo1-classLoader-parent-parent: null
}
}

从以上的①、②、③可以看出,启动类加载器是扩展类加载器的父类,扩展类加载器是应用加载器的父类。

打印出的Object的类加载器为null,表示它是根加载器,即启动类加载器。

4、类加载器的双亲委派机制

  某个特定的类加载器在接到加载类的请求时,首先将加载任务委托给父类加载器,依次递归,若父类加载器可以完成类加载任务,就成功返回;只有父类加载器无法完成此加载任务时,才自己去加载。 

  这样设计为了保证java的一种安全特性--------沙箱机制(防止恶意代码对java的破坏);

  例:若自己新建了一个String类(classpath路径下),则类加载器是应用程序类加载器;加载的时候程序类加载器委托父类加载器加载(即扩展类加载器),扩展类加载器去委托启动类加载器去加载,rt.jar中有String.class文件,则会加载这个Class,不会加载自己新建的那个String类。

JVM 类加载器 (二)的更多相关文章

  1. 深入JVM类加载器机制,值得你收藏

    先来一道题,试试水平 public static void main(String[] args) { ClassLoader c1 = ClassloaderStudy.class.getClass ...

  2. JVM类加载器的分类

    类加载器的分类 JVM支持两种类型的类加载器,分别为引导类加载器(Bootstrap ClassLoader)和自定义类加载器(User-Defined ClassLoader). 从概念上来讲,自定 ...

  3. (二十七)JVM类加载器机制与类加载过程

    一.Java虚拟机启动.加载类过程分析 下面我将定义一个非常简单的java程序并运行它,来逐步分析java虚拟机启动的过程. package org.luanlouis.jvm.load; impor ...

  4. JVM类加载器工作流程

    类加载器 classloader:谈到类加载,不得不提的就是负责此项工作的类加载器classloader,classloader的职责是将Java源文件编译后的字节码文件加载到内存中去执行. 类加载至 ...

  5. JVM 类加载器命名空间深度解析与实例分析

    一.创建Sample 1.创建实例 public class MyPerson { private MyPerson myPerson; public void setMyPerson(Object ...

  6. JVM 类加载器ClassLoader源码学习笔记

    类加载 在Java代码中,类型的加载.连接与初始化过程都是在程序运行期间完成的. 类型可以是Class,Interface, 枚举等. Java虚拟机与程序的生命周期 在如下几种情况下,Java虚拟机 ...

  7. 深入理解Java类加载器(二):线程上下文类加载器

    摘要: 博文<深入理解Java类加载器(一):Java类加载原理解析>提到的类加载器的双亲委派模型并不是一个强制性的约束模型,而是Java设计者推荐给开发者的类加载器的实现方式.在Java ...

  8. 【java虚拟机系列】JVM类加载器与ClassNotFoundException和NoClassDefFoundError

    在我们日常的项目开发中,会经常碰到ClassNotFoundException和NoClassDefFoundError这两种异常,对于经验足够的工程师而言,可能很轻松的就可以解决,但是却不一定明白为 ...

  9. jvm类加载器以及双亲委派

    首先来了解几个概念: 类加载: 概念:虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验--转换解析--初始化,最终形成能被java虚拟机直接使用的java类型,就是jvm的类加载机制. ...

随机推荐

  1. 吴裕雄 python oracle检索数据(1)

    import cx_Oracle conn = cx_Oracle.connect("scott/admin@localhost:1521/ORCL")cursor = conn. ...

  2. MATLAB GUI图片添加背景

    global im [filename,pathname]=uigetfile('*.jpg','输入图片'); file=strcat(pathname,filename); im=imread(f ...

  3. Python教程_简介2

    人生苦短,我用Python--Life is short,you need Python. https://www.bilibili.com/video/av14184325/?p=101 Pytho ...

  4. flex-grow

    [flex-grow] 指定宽度所占比,如一个flex中有三个item,这三个item的flex-grow均为1,则每个item占比为33.33%,如果一个是1,两个为2,则占比为20%,40%,40 ...

  5. appium多机并行测试

    在实际应用中需要对多个机型并行测试,节省时间 多机测试的思路 启动多个appium server与多台机器交互(android和ios均可)   注意:一定要使用node安装appium的命令行,使用 ...

  6. sqlserver 数据库迁移

    参考 https://blog.csdn.net/wuzhanwen/article/details/77449229 一.连接本地数据库引擎 新建一个数据库,如:rbrbsoft 二.连接远程数据库 ...

  7. SQLdeveloper换成windows主题后不显示的情况

    这几天因为换电脑需要重新安装数据库, 因为换成了64位系统, 原先的oracle数据库也换成了64位, 但是plsql还是要用32位的, 经过深思熟虑也没装, 请教了一个同学改用oracle自带的sq ...

  8. 每月IT摘录201811

    技术 1.打牢基础,从会使用-了解原理-了解思想一步一步来,最怕基础很弱但却以什么都用过为荣的人,这样的人我招进来也只是初级而已,工作年限再多也没有用.少林里面,有功和拳之分,如蛇拳猴拳是拳,马步功石 ...

  9. VS2013 warning C4018 "<” 有符号/无符号不匹配

    1, VS2013 warning C4018 "<” 有符号/无符号不匹配" 警告 出错代码: void show(const vector<int>& ...

  10. ELK Deployed

    Enviroment prepare rpm -qa | grep java wget http://download.oracle.com/otn-pub/java/jdk/8u171-b11/51 ...