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编程基础——数组和二维数组 摘要:本文主要对数组和二维数组进行简要介绍. 数组 定义 数组可以理解成保存一组数的容器,而变量可以理解为保存一个数的容器. 数组是一种引用类型,用于保存一组相同类 ...
随机推荐
- 通过get方法的方式获取配置项信息
这种写法比其他的方法好的一点是,当你需要修改参数名或者参数值的时候,只需要改一个地方就可以了,其他地方根本不用动,面向接口编程. eureka-server.properties archaius.d ...
- pyinstaller进行打包exe文件
百度直接pip安装,报错 下载离线文件报错. 百度了一下:还真好使 Python生成可执行文件主要有三种方法,利用py2exe,pyInstaller或cx_Freeze. 这里选择pyinstall ...
- go 自定义http.Client - 动态修改请求Body
前言 在对接Alexa Smart Home时,有的请求Payload中需要传入Access Token,但是这个Token是由OAuth2 Client管理的,封装Payload时并不知道Acces ...
- 9.2 k8s结合Jenkins与gitlab实现代码升级与回滚
1.部署Jenkins 1.1 安装jdk # apt 安装jdk11 apt install openjdk-11-jdk # 查看 root@jenkins:~# java -version op ...
- Codeforces 526G - Spiders Evil Plan(长链剖分+直径+找性质)
Codeforces 题目传送门 & 洛谷题目传送门 %%%%% 这题也太神了吧 storz 57072 %%%%% 首先容易注意到我们选择的这 \(y\) 条路径的端点一定是叶子节点,否则我 ...
- Ubuntu 彻底卸载 MySQL 数据库
Ubuntu 18.04 彻底卸载MySQL 5.7.31 1. 查看MySQL的依赖项 dpkg --list|grep mysql 2. 卸载 mysql-common sudo apt remo ...
- [R] 添加误差棒的分组折线图:geom_path: Each group consists of only one observation. Do you need to adjust the...
想做一个简单的分组折线图,并添加误差棒,类似下面这样的: 用ggplot似乎很简单就能实现:ggplot+geom_errorbar+geom_line+geom_point,重点在于计算误差棒. 还 ...
- Oracle、MySQL关机操作步骤
一.Oracle数据库单机关机(eg:LEAP系统) 先关闭使用数据库的应用系统,再关闭数据库 关闭数据库执行以下命令 1.关闭Oracle数据库监听器:(使用操作系统下管理Oracle的账户,关闭监 ...
- SonarQube的部分规则探讨
引言:为了更好的使项目代码规范化,减少Bug的出现,因此最近引入了SonarQube来帮助检测代码问题,这里就分享部分有趣的规则. 注:因为保密原则,文章贴出来的代码都是我按照格式仿写的,并非公司源码 ...
- MediaPlayer详解
[1]MediaPlayer 详细使用细则 [2]MediaPlayer使用详解_为新手准备 [3]MediaPlayer 概览