java 编程基础 类加载器
什么是类加载器
细说类加载机制
三种类加载器:
- Bootstrap ClassLoader: 根类加载器, 平台加载器的父加载器
- Platform ClassLoader: 平台类加载器,系统加载器的父加载器。在Java8和之前,这个加载器应该叫做扩展加载器(ExtClassLoader)
- System ClassLoader: 系统类加载器。
1,根类加载器:
2,平台类加载器:
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);
}
}

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

自定义类加载器
- (1), loadClass(String name, boolean resove) 该方法是ClassLoader的入口点,根据指定名称来加载。系统就是调用ClassLoader的该方法来获取指定类Class对象。
- (2), findClass(String name): 根据指定名称来加载类。
- 用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 编程基础 类加载器的更多相关文章
- Java中的类加载器--Class loader
学习一下Java中的类加载器,这个是比较底层的东西,好好学习.理解一下. 一.类加载器的介绍 1.类加载器:就是加载类的工具,在java程序中用到一个类,java虚拟机首先要把这个类的字节码加载到内 ...
- 黑马程序员——【Java高新技术】——类加载器
---------- android培训.java培训.期待与您交流! ---------- 一.概述 (一)类加载器(class loader) 用来动态加载Java类的工具,它本身也是Java类. ...
- Java中的类加载器
转载:http://blog.csdn.net/zhangjg_blog/article/details/16102131 从java的动态性到类加载机制 我们知道,Java是一种动态语言.那么怎 ...
- Java中的类加载器以及Tomcat的类加载机制
在加载阶段,虚拟机需要完成以下三件事情: 1.通过一个类的全限定名来获取其定义的二进制字节流. 2.将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构. 3.在Java堆中生成一个代表这个类 ...
- Java入门——(1)Java编程基础
Java入门--(1)Java编程基础 第二章 Java编程基础 JAVA 代码的基本格式: 修饰符 class 类名{ 程序代码 } 2.1关键字:赋予了特殊含义的单词. 2.2标识符: ...
- Java开发知识之Java编程基础
Java开发知识之Java编程基础 一丶Java的基础语法 每个语言都有自己的语法规范.例如C++ 入口点是main. 我们按照特定格式编写即可. Java也不例外. Java程序的语法规范就是 Ja ...
- java编程基础二进制
0.java编程基础 01.二进制(原码,反码,补码) 02.位运算 03.移位运算符 二进制 原码,反码,补码 1.基本概念 二进制是逢2进位的进位制,0,1是基本算符. 现在的电子计算机技术全部使 ...
- Java编程基础-面向对象(中)
本章承接Java编程基础-面向对象(上)一文. 一.static关键字 在java中,定义了一个static关键字,它用于修饰类的成员,如成员变量.成员方法以及代码块等,被static修饰的成员具备一 ...
- Java编程基础——数组和二维数组
Java编程基础——数组和二维数组 摘要:本文主要对数组和二维数组进行简要介绍. 数组 定义 数组可以理解成保存一组数的容器,而变量可以理解为保存一个数的容器. 数组是一种引用类型,用于保存一组相同类 ...
随机推荐
- Python的数据解析
- P6072 『MdOI R1』Path
考虑我们有这样操作. 我们只要维护两点在子树内和两点在子树外的异或和即可. 前者可以类似于线段树合并的trie树合并. 后者有两种做法: 一种是把dfn序翻倍:然后子树补变成了一个区间最大异或问题,可 ...
- 洛谷 P6199 - [EER1]河童重工(点分治+虚树)
洛谷题面传送门 神仙题. 首先看到这样两棵树的题目,我们肯定会往动态树分治的方向考虑.考虑每次找出 \(T_2\) 的重心进行点分治.然后考虑跨过分治中心的点对之间的连边情况.由于连边边权与两棵树都有 ...
- sigma网格中水平压力梯度误差及其修正
1.水平梯度误差产生 sigma坐标系下,笛卡尔坐标内水平梯度项对应形式为 \[\begin{equation} \left. \frac{\partial }{\partial x} \right| ...
- python11文件读写模块
将文件的打开和关闭,交给上下文管理工具with去实现. def read_file(): """ 读取文件 :return: """ fil ...
- 执行脚本source 和 . 和sh 的区别是什么
"source"和"."的功能是一样的,可以调用脚本,并将脚本里的函数也传递到当前的脚本或者解释器中,即不会开启新的bash而是在当前bash中运行. &quo ...
- Linux—export命令查看、修改用户环境变量
Linux export 命令用于设置或显示环境变量. 在 shell 中执行程序时,shell 会提供一组环境变量. export 可新增,修改或删除环境变量,供后续执行的程序使用. export ...
- day15 内置函数和模块
day15 内置函数和模块 1.三元表达式 代码如下: x = 1 y = 2 res = 'ok' if x > y else 'no' print(res) 输出结果:no 2.内置函数:重 ...
- 如何删除苹果电脑垃圾文件-7个高级技巧释放大量苹果Mac
硬盘空间用尽是一件很让人头疼的事情,尤其是MacBook Air等设备上的固态硬盘可用的储存空间很少.下面[微IT]为大家介绍7个高级技巧来释放大量的硬盘空间,当然这些高级技巧更改了系统功能和文件,必 ...
- 【1】Embarrassingly Parallel(易并行计算问题)
1.什么是Embarrassingly Parallel(易并行计算问题) 易并行计算问题:A computation that can be divided into a number of co ...