中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java中如何實現動態加載jar和class文件

發布時間:2021-08-05 14:15:15 來源:億速云 閱讀:350 作者:小新 欄目:編程語言

小編給大家分享一下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和class文件”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

梁平县| 尉犁县| 贵南县| 普洱| 泰顺县| 瓦房店市| 清流县| 潼南县| 乐亭县| 开原市| 临朐县| 和平县| 正安县| 乐陵市| 仁怀市| 周口市| 余江县| 安溪县| 辽中县| 宜兰市| 东兴市| 肥西县| 加查县| 乐安县| 小金县| 大宁县| 平武县| 武功县| 呼和浩特市| 廊坊市| 平舆县| 宜州市| 西和县| 贡山| 西乌珠穆沁旗| 抚顺县| 清镇市| 丁青县| 竹北市| 蒲城县| 大埔县|