Java 动态加载jar和class文件
2023-09-11 14:22:55 时间
JAVA中类文件加载是动态的。也就是说当我们用到的时候才会去加载,如果不用的话,就不会去加载我们的类。
JAVA为我们提供了两种动态机制。第一种是隐式机制。第二种是显示机制。如下:
两种方法:
- 隐式机制 :new一个对象 + 调用类的静态方法
- 显式机制 :
- 由 java.lang.Class的forName()方法加载
- 由 java.lang.ClassLoader的loadClass()方法加载
1、Class.forName
Class.forName()方法具有两个形式:
- public static Class forName(String className)
- public static Class forName(String className, boolean initialize,ClassLoader loader)
参数说明:
- className - 所需类的完全限定名 (必须包含包名,否则出错!)
- initialize - 是否必须初始化类 (静态代码块的初始化)
- loader - 用于加载类的类加载器
调用只有一个参数的forName()方法等效于 Class.forName(className, true, loader)。
这两个方法,最后都要连接到原生方法forName0().
而三个参数的forName(),最后调用的是: forName0(name, initialize, loader);
不管使用的是new 來实例化某个类、或是使用只有一个参数的Class.forName()方法,内部都隐含了“载入类 + 运行静态代码块”的步骤。
而使用具有三个参数的Class.forName()方法时,如果第二个参数为false,那么类加载器只会加载类,而不会初始化静态代码块,只有当实例化这个类的时候,静态代码块才会被初始化,静态代码块是在类第一次实例化的时候才初始化的。
2、java.lang.ClassLoader
ClassLoader就是用来Load Class的,当一个Class被加载的时候,这个Class所引用到的所有Class也会被加载,而且这种加载是递归的,也就是说,如果A引用到B,B 引用到C,那么当A被加载的时候,B也会被加载,而B被加载的时候,C也会加载。如此递归直到所有需要的Class都加载好。
package com.demo.test;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
public class DynamicLoadDemo {
enum FileType {
JAR, CLASS, OTHER
}
static class MyClassLoader extends ClassLoader {
public synchronized Class<?> loadClass(String name, File file) throws FileNotFoundException {
Class<?> cls = findLoadedClass(name);
if(cls != null) {
return cls;
}
FileInputStream fis = new FileInputStream(file);
ByteArrayOutputStream baos = new ByteArrayOutputStream();
byte[] buffer = new byte[1024];
int len;
try {
while (true) {
len = fis.read(buffer);
if (len == -1) {
break;
}
baos.write(buffer, 0, len);
}
//FileInputStream的flush是空操作,因为flush的作用是把缓存中的东西写入实体(硬盘或网络流)中,这里没有这种必要所以为空
//baos.flush();
byte[] data = baos.toByteArray();
return defineClass(null, data, 0, data.length);
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
baos.close();
} catch (IOException e) {
e.printStackTrace();
}
try {
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return null;
}
}
public static void main(String[] args) {
String className = "com.demo.test.HelloWorld";
String paths[] = { "HelloWorld.jar", "HelloWorld.class" };
for (String path : paths) {
String lowerPath = path.toLowerCase();
FileType fileType = FileType.OTHER;
if (lowerPath.endsWith(".jar") || lowerPath.endsWith(".zip")) {
fileType = FileType.JAR;
} else if (lowerPath.endsWith(".class")) {
fileType = FileType.CLASS;
}
if (fileType == FileType.OTHER) {
return;
}
File file = new File(path);
if (!file.exists()) {
return;
}
try {
URL url = file.toURI().toURL();
System.out.println(url.toString());
Class<?> cls = null;
switch (fileType) {
case JAR:
URLClassLoader classLoader = new URLClassLoader(new URL[] { url }, Thread.currentThread().getContextClassLoader());
cls = classLoader.loadClass(className);
break;
case CLASS:
MyClassLoader myClassLoader = new MyClassLoader();
cls = myClassLoader.loadClass(className, file);
break;
default:
break;
}
if (cls == null) {
return;
}
// 实例变量
Field field = cls.getDeclaredField("hello");
if (!field.isAccessible()) {
field.setAccessible(true);
}
System.out.println(field.get(cls.newInstance()));
// 调用静态不带参数方法
Method staticMethod = cls.getDeclaredMethod("sayStaticHello", null);
if (!staticMethod.isAccessible()) {
staticMethod.setAccessible(true);
}
// 如果函数的返回值是void,就会返回null
staticMethod.invoke(cls, null);
// 实例带参数方法方法
Method method = cls.getDeclaredMethod("say", String.class);
if (!method.isAccessible()) {
method.setAccessible(true);
}
Object ret = method.invoke(cls.newInstance(), "hello world");
System.out.println(ret);
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
} catch (FileNotFoundException e) {
e.printStackTrace();
}
}
}
}
相关文章
- java高级用法之:在JNA中将本地方法映射到JAVA代码中
- 你所不知道的库存超限做法 服务器一般达到多少qps比较好[转] JAVA格物致知基础篇:你所不知道的返回码 深入了解EntityFramework Core 2.1延迟加载(Lazy Loading) EntityFramework 6.x和EntityFramework Core关系映射中导航属性必须是public? 藏在正则表达式里的陷阱 两道面试题,带你解析Java类加载机制
- windows注册java的jar包为服务
- Idea:Java文件打包可执行的JAR文件,并指定JDK编译版本
- Java反编译生成java文件
- Java 内存溢出(java.lang.OutOfMemoryError)的常见情况和处理方式总结
- 34 异常机制 异常体系结构 Java把异常当做对象来处理 并定义一个基类java.lang.Throwable作为所有异常的超类 Error Exception
- 第七节:详细讲解Java中的日期,java.util.date
- 使用java -jar运行jar包
- Java经典实例:纪元秒和本地日期时间互换
- springboot项目启动 java -jar详解
- Java基础之反射
- Java 设计模式之装饰模式,Java 装饰模式,java装饰模式和代理模式的区别
- Java学习-051-Detected both log4j-over-slf4j.jar AND bound slf4j-log4j12.jar on the class path, preempting StackOverflowError
- Java中使用com.sun相关jar包出现编译错误,但是运行没有错误的解决方法和原因
- java中的 public protected friendly private
- java中jar命令打包一个文件夹下的所有文件
- Java_java动态编译整个项目,解决jar包找不到问题
- 『Java练习生的自我修养』java-se进阶¹ • 初识多线程
- 加载依赖的jar包在命令行编译和运行java文件
- Json——使用Json jar包实现Json字符串与Java对象或集合之间的互相转换
- java 通过ffmpeg 访问网络视频
- 【JAVA基础】不同的jar拥有相同全限定类名和不同的方法Method时NoSuchMethodError,同名类加载问题 / 双亲委派
- 真正解决问题:maven eclipse tomcat java.lang.ClassNotFoundException: org.springframework.web.context.ContextLoaderListener--转
- Java笔记3:Eclipse添加jar包