自定义ClassLoader,用于加载用户JAR包

时间:2023-03-09 15:10:40
自定义ClassLoader,用于加载用户JAR包

最近在考虑C/S结构程序的软件自动升级的实现方式,比如QQ、飞信等都自动升级的功能。

自动升级模块虽然还没有编码完成,但是思路还是比较清晰的。

自动升级过程中,升级文件的JAR包是专门加载到程序中去的,因此,自定义一个ClassLoader,用于加载用户JAR包,就非常的重要了。

应用程序ClassLoader只提供了一个public Class<?> loadClass(String name) throws ClassNotFoundException 方法,没有提供加载JAR的方法。

URLClassLoader提供了一个protected void addURL(URL url)的方法,倒是可以加载JAR包,但苦于非public的。

经测试发现,AppClassLoader是URLClassLoader的子类。因此,我们完全可以利用URLClassLoader了哦。

  1. URLClassLoader system = (URLClassLoader) ClassLoader.getSystemClassLoader();
URLClassLoader system = (URLClassLoader) ClassLoader.getSystemClassLoader();

这样,我们可以通过反射得到addURL方法,在程序中加载我们自己的JAR包了。

整个源代码如下所示:

  1. /**
  2. * Copyright (c) YMCN Team
  3. * All rights reserved.
  4. */
  5. package com.aboy.toolkit.util;
  6. import java.io.File;
  7. import java.lang.reflect.Method;
  8. import java.net.URL;
  9. import java.net.URLClassLoader;
  10. import java.util.ArrayList;
  11. import java.util.List;
  12. /**
  13. * @author obullxl
  14. *
  15. * email: obullxl@163.com  MSN: obullxl@hotmail.com  QQ: 303630027
  16. *
  17. * Blog: http://obullxl.iteye.com
  18. */
  19. public final class ClassLoaderUtil {
  20. /** URLClassLoader的addURL方法 */
  21. private static Method addURL = initAddMethod();
  22. /** 初始化方法 */
  23. private static final Method initAddMethod() {
  24. try {
  25. Method add = URLClassLoader.class
  26. .getDeclaredMethod("addURL", new Class[] { URL.class });
  27. add.setAccessible(true);
  28. return add;
  29. } catch (Exception e) {
  30. e.printStackTrace();
  31. }
  32. return null;
  33. }
  34. private static URLClassLoader system = (URLClassLoader) ClassLoader.getSystemClassLoader();
  35. /**
  36. * 循环遍历目录,找出所有的JAR包
  37. */
  38. private static final void loopFiles(File file, List<File> files) {
  39. if (file.isDirectory()) {
  40. File[] tmps = file.listFiles();
  41. for (File tmp : tmps) {
  42. loopFiles(tmp, files);
  43. }
  44. } else {
  45. if (file.getAbsolutePath().endsWith(".jar") || file.getAbsolutePath().endsWith(".zip")) {
  46. files.add(file);
  47. }
  48. }
  49. }
  50. /**
  51. * <pre>
  52. * 加载JAR文件
  53. * </pre>
  54. *
  55. * @param file
  56. */
  57. public static final void loadJarFile(File file) {
  58. try {
  59. addURL.invoke(system, new Object[] { file.toURI().toURL() });
  60. System.out.println("加载JAR包:" + file.getAbsolutePath());
  61. } catch (Exception e) {
  62. e.printStackTrace();
  63. }
  64. }
  65. /**
  66. * <pre>
  67. * 从一个目录加载所有JAR文件
  68. * </pre>
  69. *
  70. * @param path
  71. */
  72. public static final void loadJarPath(String path) {
  73. List<File> files = new ArrayList<File>();
  74. File lib = new File(path);
  75. loopFiles(lib, files);
  76. for (File file : files) {
  77. loadJarFile(file);
  78. }
  79. }
  80. }
/**
* Copyright (c) YMCN Team
* All rights reserved.
*/
package com.aboy.toolkit.util; import java.io.File;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.ArrayList;
import java.util.List; /**
* @author obullxl
*
* email: obullxl@163.com MSN: obullxl@hotmail.com QQ: 303630027
*
* Blog: http://obullxl.iteye.com
*/
public final class ClassLoaderUtil {
/** URLClassLoader的addURL方法 */
private static Method addURL = initAddMethod(); /** 初始化方法 */
private static final Method initAddMethod() {
try {
Method add = URLClassLoader.class
.getDeclaredMethod("addURL", new Class[] { URL.class });
add.setAccessible(true);
return add;
} catch (Exception e) {
e.printStackTrace();
}
return null;
} private static URLClassLoader system = (URLClassLoader) ClassLoader.getSystemClassLoader(); /**
* 循环遍历目录,找出所有的JAR包
*/
private static final void loopFiles(File file, List<File> files) {
if (file.isDirectory()) {
File[] tmps = file.listFiles();
for (File tmp : tmps) {
loopFiles(tmp, files);
}
} else {
if (file.getAbsolutePath().endsWith(".jar") || file.getAbsolutePath().endsWith(".zip")) {
files.add(file);
}
}
} /**
* <pre>
* 加载JAR文件
* </pre>
*
* @param file
*/
public static final void loadJarFile(File file) {
try {
addURL.invoke(system, new Object[] { file.toURI().toURL() });
System.out.println("加载JAR包:" + file.getAbsolutePath());
} catch (Exception e) {
e.printStackTrace();
}
} /**
* <pre>
* 从一个目录加载所有JAR文件
* </pre>
*
* @param path
*/
public static final void loadJarPath(String path) {
List<File> files = new ArrayList<File>();
File lib = new File(path);
loopFiles(lib, files);
for (File file : files) {
loadJarFile(file);
}
}
}

在程序中,只要使用上面最后两个方法,就可以加载自定义JAR包和一个目录中的所有JAR包了。

经本人测试,上面代码运行无误,能正常加载自定义JAR包。

欢迎指正~~~~~~~