Java 动态加载jar和class文件实例解析

本文研究的主要是Java 动态加载jar和class文件的相关内容,具体如下。

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 动态加载jar和class文件实例解析的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站其他相关专题,如有不足之处,欢迎留言指出。感谢朋友们对本站的支持!

您可能感兴趣的文章:

  • java动态添加外部jar包到classpath的实例详解
  • 解决Eclipse add external jars运行出现java.lang.NoClassDefFoundError的方法
(0)

相关推荐

  • 解决Eclipse add external jars运行出现java.lang.NoClassDefFoundError的方法

    最近发现一个问题,有时候对一个Android项目反复的Add jar和remove jar,发现编译可以通过,但是运行起来当应用到外部jar的对象时,会抛出java.lang.NoClassDefFoundError异常.导致程序奔溃. 查看项目属性,发现java build path里比正常的项目少了Android Dependencies这一项: 通过和丢失之前的项目进行比对,发现差别在于工程根目录下的.classpath文件,<classpathentry exported="tru

  • java动态添加外部jar包到classpath的实例详解

    java动态添加外部jar包到classpath的实例详解 前言: 在项目开发过程中我们有时候需要动态的添加外部jar包,但是具体的业务需求还没有遇到过,因为如果动态添加外部jar包后,我们就需要修改业务代码,而修改代码就需要重新启动服务,那样好像就没有必要动态添加外部jar包了,怎么样才能不重新启动服务器就可以使用最新的代码我没有找到方法,如果各位知道的话给我点建议,回归主题,实现动态添加外部jar包到classpath的方法如下: String beanClassName = "com.dy

  • Java 动态加载jar和class文件实例解析

    本文研究的主要是Java 动态加载jar和class文件的相关内容,具体如下. JAVA中类文件加载是动态的.也就是说当我们用到的时候才会去加载,如果不用的话,就不会去加载我们的类. JAVA为我们提供了两种动态机制.第一种是隐式机制.第二种是显示机制.如下: 两种方法: 隐式机制 :new一个对象 + 调用类的静态方法 显式机制 :由 java.lang.Class的forName()方法加载 由 java.lang.ClassLoader的loadClass()方法加载 1.Class.fo

  • java 动态加载的实现代码

    java 动态加载的实现代码 Java动态加载类的意义和目的: Java动态加载类主要是为了不改变主程序代码,通过修改配置文件就可以操作不同的对象执行不同的功能.主要有利于系统的扩展,例如当我要改变一个功能,只需要做一个类,然后编写相应的功能,通过配置文件就可以使用新的功能,不需要修改系统的任何地方,只需要添加一个类:充分实现了松散耦合.满足了开闭原则(对修改关闭,对添加或删除开放): public abstract class AbstractAction { public abstract

  • Java动态加载类示例详解

    在讲解动态加载类之前呢,我们先弄清楚为什么要动态加载类,静态加载不行吗?我们可以看下面的实例: 我在文件夹里写了Office.java 类和 Word.java类,如下: Office.java class Office{ public static void main(String[] args){ if(args[0].equals("Word")){ Word w = new Word(); w.start(); } if(args[0].equals("Excel&q

  • 如何实现JavaScript动态加载CSS和JS文件

    项目中需要用到动态加载CSS 文件,整理了一下,顺便融合了动态加载JS 的功能写成了一个对象,先上代码: var dynamicLoading = { css: function(path){ if(!path || path.length === 0){ throw new Error('argument "path" is required !'); } var head = document.getElementsByTagName('head')[0]; var link =

  • 动态加载js、css的实例代码

    一.原生js: /** * 加载js和css文件 * @param jsonData.path 前缀路径 * @param jsonData.url 需要加载的js路径或css路径 * @param jsonData.type 需要加载的类型 js或css */ function loadWriteFiles(jsonData) { jsonData.path = jsonData.path != undefined ? jsonData.path : ""; if(jsonData.

  • bootstrap里bootstrap动态加载下拉框的实例讲解

    实例如下所示: //引入的包 <!-- bootstrap --> <link rel="stylesheet" type="text/css" href="map/plug-in/scripts/bootstrap/bootstrap.min.css" rel="external nofollow" /> <link rel="stylesheet" type="t

  • Android 中动态加载.jar的实现步骤

    首先第一个是 jar 文件的制作,Java 里面直接把 .class 文件打包到 .jar 文件里面就可以了,但是 Android 的 Dalvik VM 是不认 Java 的 byte code 的,所以不能直接这么打包,而要用 dx 工具转成 Dalvik byte code 才可以.当然,dx 工具转了之后,jar 包里面就不 是 .class 文件了,而是 .dex 文件. 第二个是,Android 里面虽然也提供了 URLClassLoader 的实现,但是并不能用.要动态加载其它类,

  • 详解Java动态加载数据库驱动

    问题背景 在同一套系统中,要支持连接访问各种流行的数据库,以及同一数据库的不同版本,例如,oracle9i.oracle10g.oracle11g.oracle12c.sqlserver2000.sqlserver2005.sqlserver2008.sqlserver2012等,其中就会碰到一些问题,就是不同的数据库,数据库驱动肯定不同,对于这个问题到好解决,只需要将相应的驱动加入即可:然而对于同种数据库,不同版本时,而且不同版本的数据库驱动不仅不兼容,同时存在还会出现冲突,例如,能满足sql

  • 异步动态加载js与css文件的js代码

    jquery动态加载css,js文件方法简单很, 例 方法1: 代码如下 复制代码 代码如下: $.getscript("test.js"); 方法2: 代码如下 复制代码 代码如下: function loadjs(file){ var head = $('head').remove('#loadscript'); $("<scri"+"pt>"+"</scr"+"ipt>").a

  • java动态加载插件化编程详解

    前言 对于java程序员来说,插件化是一件很酷的功能,小二有幸在工作中实现了此功能. 背景: 需要将mysql的数据通过canal同步至kafka/mysql/hdfs等 实现 /** * Created by shengjk1 on 2017/12/11 */ public class PluginManager { private final static Logger logger = LoggerFactory.getLogger(SendMessageFactory.class); p

随机推荐