什么是类加载器

类加载器负责将class文件(可能在磁盘上,也可能在网络上)加载到内存中,并为之生成对应的java.lang.Class对象。Java开发中无须过分关心类加载机制,但所有的编程人员都应该了解其工作机制,明白如何做才能让其更好地满足我们的编程需要。
 
 

细说类加载机制

1,一个类只会被加载一次:
类加载器负责加载所有的类,系统为所有被载入内存中的类生成java.lang.Class实例。只要一个类被载入JVM中,同一个类就不会被再次载入了。现在的问题是,怎么样才算"同一个类"?
我们的对象实例是不是都有一个hashCode来作为其唯一的标识?在JVM中一个类也有其对应的唯一标识。这个唯一标识使用全限定类名(包名全路径+类名)来作为标识的。
 
 

三种类加载器:

JVM 启动时,会形成由三个类加载器组成的初始类加载器层次结构。
  • Bootstrap ClassLoader: 根类加载器, 平台加载器的父加载器
  • Platform ClassLoader: 平台类加载器,系统加载器的父加载器。在Java8和之前,这个加载器应该叫做扩展加载器(ExtClassLoader)
  • System ClassLoader: 系统类加载器。
 

1,根类加载器:

Bootstrap ClassLoader被称为引导加载器,有些书上也叫做原始类加载器或者根类加载器。它负责加载Java的核心类。这个加载器是由JVM自己实现的(C/C++)实现,所以我们没办法直接使用到对应的类。
这个加载器加载的类的路径是jdk安装目录下面对应jre/lib目录下的核心库

2,平台类加载器:

如果是java8以及之前,这个对应的是扩展加载器。加载的是jre/lib/ext目录下的扩展包。而在java8之后,平台类加载器只是为了向后兼容而保留,而不会加载任何东西了。

3,系统加载器:

加载的是对应的入口程序所在的包。
编写代码查看类加载器路径:

package com.vgxit.classloeader;
import java.io.IOException;
import java.net.URL;
import java.net.URLDecoder;
import java.util.Enumeration;
public class ClassLoaderPropTest {
public static void main(String[] args) throws IOException {
//首先获取对应的系统加载器
ClassLoader systemLoader = ClassLoader.getSystemClassLoader();
System.out.println("系统加载器:" + systemLoader);
//打印系统加载器的路径
Enumeration<URL> enumeration = systemLoader.getResources("");
while (enumeration.hasMoreElements()) {
System.out.println(URLDecoder.decode(enumeration.nextElement().toString(), "utf-8"));
}
//获取系统加载器的父加载器(平台类加载器)
ClassLoader platformLoader = systemLoader.getParent();
System.out.println("平台类加载器" + platformLoader);
//打印一下平台类加载器的路径
Enumeration<URL> enumeration1 = platformLoader.getResources("");
while (enumeration1.hasMoreElements()) {
System.out.println(URLDecoder.decode(enumeration1.nextElement().toString(), "utf-8"));
}
//获取根加载器
ClassLoader bootstrapLoader = platformLoader.getParent();
System.out.println("根加载器是:" + bootstrapLoader);
}
}
可以看出:
系统加载器AppClassLoader的加载路径就是当前程序的运行的路径 。
平台类加载器是PlatformClassLoader。如果是在java8下面运行打印出来是ExtClassLoader。平台类加载器的路径没有(这个和Java8是有区别的)。
跟加载器返回的是null。这个是因为根加载器不是用java实现的,并没有继承我们的ClassLoader抽象类。所以说我们获取不到。但是实际上根类加载器就是我们平台类加载器的父加载器。

三种类加载机制:

  • 全盘负责:所谓全盘负责,就是当一个类加载器负责加载某C1ass时,该class所依赖的和引用的其他class也将由该类加载器负责载入,除非显式使用另外一个类加载器来载入
  • 父类委托:所谓父类委托,是先让parent(父)类加载器试图加载该class,只有在父类加载器无法加载该类时才尝试从自己的类路径中加载该类。
  • 缓存机制:缓存机制将会保证所有加载过的C1ass都会被缓存,当程序中需要使用某个class时,类加载器先从缓存区中搜寻该C1ass,只有当缓存区中不存在该Class对象时,系统才会读取该类对应的二进制数据,并将其转换成C1ass对象,存入缓存区中。这就是为什么修改了代码后,必须重新启动JVM程序所做的修改才会生效的原因。

类加载器工作步骤

自定义类加载器

JVM中除根类加载器之外的所有类加载器都是ClassLoader的子类的实例,开发者可以通过扩展ClassLoader的子类,并重写该ClassLoader所包含的方法来实现自定义的加载器。
 
ClassLoader 类有如下两个关键方法:
  • (1), loadClass(String name, boolean resove) 该方法是ClassLoader的入口点,根据指定名称来加载。系统就是调用ClassLoader的该方法来获取指定类Class对象。
  • (2), findClass(String name): 根据指定名称来加载类。
 
如果需要实现自定义的ClassLoader,则可以重写这两个方法来实现,通常推荐重写 findClass方法,而不是重写loadClass方法。loadClass方法的执行步骤如下:
  • 用findLoadedClass(String name)来检查是否己经加载类,如果已经加载则直接返回。
  • 在父类加载器上调用loadClass方法。 果父类加载器为null,则使用根类加载器来加载
  • 调用findClass(String)方法查找类
    package com.zmd.myclassloader;
    
    import java.io.*;
    import java.lang.reflect.InvocationTargetException;
    import java.lang.reflect.Method; /**
    * @ClassName MyClassLoader
    * @projectName: object1
    * @author: Zhangmingda
    * @description: 自定义加载器,继承ClassLoader类
    * date: 2021/5/15.
    */
    public class MyClassLoader extends ClassLoader{
    /**
    * @param name 重写findClass 方法 name为类名称
    * @return 返回加载好的类
    * @throws ClassNotFoundException 异常
    */
    @Override
    protected Class<?> findClass(String name) throws ClassNotFoundException {
    //定义好要返回的类
    Class<?> cls = null;
    //替换类名路径.为目录/
    String filePath = name.replace('.','/');
    String javaFileName = filePath + ".java";
    String classFileName = filePath + ".class";
    File javaFile = new File(javaFileName);
    File classFile = new File(classFileName);
    System.out.println(javaFileName);
    if (!javaFile.exists() && !classFile.exists()){
    throw new ClassNotFoundException("类:" + name + "不存在");
    }else if (javaFile.exists()){
    if (! classFile.exists() || classFile.lastModified() < javaFile.lastModified()){
    try {
    if (! complie(javaFile) || ! classFile.exists()){
    throw new ClassNotFoundException("类:" + name + "编译失败");
    }
    } catch (IOException e) {
    e.printStackTrace();
    } catch (InterruptedException e) {
    e.printStackTrace();
    }
    //编译成功,读取二进制文件,加载到内存中,转化成对应的Class对象
    try (FileInputStream fileInputStream = new FileInputStream(classFile)) {
    //定义存储二进制数据的byte数组
    byte[] classBytes = new byte[ (int) classFile.length()];
    int readLen = fileInputStream.read(classBytes);
    //转化成Class对象
    cls = defineClass(name,classBytes,0, readLen);
    } catch (FileNotFoundException e) {
    e.printStackTrace();
    } catch (IOException e) {
    e.printStackTrace();
    }
    }
    }
    return cls;
    }
    /**
    * 编译动作
    */
    private boolean complie(File javaFile) throws IOException, InterruptedException {
    System.out.println("开始编译:" + javaFile.getPath());
    //操作系统执行编译动作
    try {
    Process process = Runtime.getRuntime().exec("javac " + javaFile);
    process.waitFor();
    int result = process.exitValue();
    System.out.println("result:" + result);
    return result == 0;
    }catch (InterruptedException e) {e .printStackTrace();}
    //等待当前进程完成
    return false;
    }
    /**
    * 入口main方法测试
    */
    public static void main(String[] args) throws ClassNotFoundException, NoSuchMethodException, InvocationTargetException, IllegalAccessException {
    if (args.length < 1){
    System.err.println("没有指定目标类");
    System.err.println(" java com.zmd.myclassloader.MyClassLoader com.zmd.myclassloader.Test  hh haa");
    return;
    }
    String className = args[0];
    String[] runArgs = new String[args.length -1];
    System.out.println("className:"+ className);
    System.arraycopy(args,1, runArgs,0, runArgs.length);
    //构建加载器,加载类
    MyClassLoader myClassLoader = new MyClassLoader();
    Class<?> cls = myClassLoader.loadClass(className);
    //通过反射执行
    Method method = cls.getMethod("main",runArgs.getClass());
    method.invoke(null,new Object[]{runArgs});
    }
    }

    Test 类

    package com.zmd.myclassloader;
    
    /**
    * @ClassName Test
    * @projectName: object1
    * @author: Zhangmingda
    * @description: XXX
    * date: 2021/5/15.
    */
    public class Test {
    public static void main(String[] args) {
    System.out.println("This is Test class ,args :" + args.toString() );
    }
    }

java 编程基础 类加载器的更多相关文章

  1. Java中的类加载器--Class loader

    学习一下Java中的类加载器,这个是比较底层的东西,好好学习.理解一下.  一.类加载器的介绍 1.类加载器:就是加载类的工具,在java程序中用到一个类,java虚拟机首先要把这个类的字节码加载到内 ...

  2. 黑马程序员——【Java高新技术】——类加载器

    ---------- android培训.java培训.期待与您交流! ---------- 一.概述 (一)类加载器(class loader) 用来动态加载Java类的工具,它本身也是Java类. ...

  3. Java中的类加载器

    转载:http://blog.csdn.net/zhangjg_blog/article/details/16102131 从java的动态性到类加载机制   我们知道,Java是一种动态语言.那么怎 ...

  4. Java中的类加载器以及Tomcat的类加载机制

    在加载阶段,虚拟机需要完成以下三件事情: 1.通过一个类的全限定名来获取其定义的二进制字节流. 2.将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构. 3.在Java堆中生成一个代表这个类 ...

  5. Java入门——(1)Java编程基础

    Java入门--(1)Java编程基础 第二章 Java编程基础   JAVA 代码的基本格式: 修饰符 class 类名{ 程序代码 }   2.1关键字:赋予了特殊含义的单词.   2.2标识符: ...

  6. Java开发知识之Java编程基础

    Java开发知识之Java编程基础 一丶Java的基础语法 每个语言都有自己的语法规范.例如C++ 入口点是main. 我们按照特定格式编写即可. Java也不例外. Java程序的语法规范就是 Ja ...

  7. java编程基础二进制

    0.java编程基础 01.二进制(原码,反码,补码) 02.位运算 03.移位运算符 二进制 原码,反码,补码 1.基本概念 二进制是逢2进位的进位制,0,1是基本算符. 现在的电子计算机技术全部使 ...

  8. Java编程基础-面向对象(中)

    本章承接Java编程基础-面向对象(上)一文. 一.static关键字 在java中,定义了一个static关键字,它用于修饰类的成员,如成员变量.成员方法以及代码块等,被static修饰的成员具备一 ...

  9. Java编程基础——数组和二维数组

    Java编程基础——数组和二维数组 摘要:本文主要对数组和二维数组进行简要介绍. 数组 定义 数组可以理解成保存一组数的容器,而变量可以理解为保存一个数的容器. 数组是一种引用类型,用于保存一组相同类 ...

随机推荐

  1. CF1578J Just Kingdom

    考虑一个点被填满则他需要从其父亲得到\(q_u = \sum_{v = u\ or\ v \in son_u}m_v\) 那么考虑如何这样操作. 我当时world final做的时候,是从上往下遍历, ...

  2. [NOIP2017 提高组] 逛公园

    考虑先做一个\(dp\),考虑正反建图,然后按0边拓扑,然后按1到这里的最小距离排序,然后扩展这个\(f_{i,j}\),即多了\(j\)的代价的方案数.

  3. Codeforces 917C - Pollywog(状压 dp+矩阵优化)

    UPD 2021.4.9:修了个 typo,为啥写题解老出现 typo 啊( Codeforces 题目传送门 & 洛谷题目传送门 这是一道 *2900 的 D1C,不过还是被我想出来了 u1 ...

  4. 详细解析Thinkphp5.1源码执行入口文件index.php运行过程

    详细解析Thinkphp5.1源码执行入口文件index.php运行过程 运行了public目录下的index.php文件后,tp的运行整个运行过程的解析 入口文件index.php代码如下: < ...

  5. PHP 获取两个日期相差多少年,多少月,多少天,多少小时,并填充数组

    PHP 获取两个日期相差多少年,多少月,多少天,多少小时,并填充数组 <?php /** * 获取两个日期相差多少年,多少月,多少天,多少小时,并填充数组 * @param [type] $st ...

  6. cmd到指定目录并执行命令 mysql到bin目录并执行命令 cmd bat进入指定文件夹中并执行命令

    其实就一条命令:(保存为bat格式,注意:有两个and希腊字母 && )cmd /k "cd /d Your ProjectPath&&Your CMD co ...

  7. 日常Java 2021/9/19

    Math类方法 package m; public class m { public static void main(String args[]) { //计算平方根 System.out.prin ...

  8. ES5中改变this指向的三种方法

    ES5中提供了三种改变函数中this指针指向的方法,分别如下 1.call() var obj = {username:"孙悟空"}; //没有任何修饰的调用函数,函数中的this ...

  9. 链式栈——Java实现

    1 package struct; 2 3 //接口 4 interface ILinkStack{ 5 //栈中元素个数(栈大小) 6 int size(); 7 //取栈顶元素 8 Object ...

  10. AFNetworking 网络错误提示data转换字符串

    AFN在进行网络交互时,有时候会碰到返回502.500.404的时候.后台的总需要你配合他查出问题所在.但是AFN在返回数据序列化时解析错误只会转成NSData类型的数据,如果直接扔给后台Data的数 ...