[转载] Java中动态加载jar文件和class文件

时间:2023-04-30 11:16:44

转载自http://blog.csdn.net/mousebaby808/article/details/31788325

概述

  诸如tomcat这样的服务器,在启动的时候会加载应用程序中lib目录下的jar文件以及classes目录下的class文件,另外像spring这类框架,也可以根据指定的路径扫描并加载指定的类文件,这个技术可以实现一个容器,容纳各类不同的子应用。
  Java类由于需要加载和编译字节码,动态加载class文件较为麻烦,不像C加载动态链接库只要一个文件名就可以搞定,但JDK仍提供了一整套方法来动态加载jar文件和class文件。

动态加载jar文件

  1. // 系统类库路径
  2. File libPath = new File(jar文件所在路径);
  3. // 获取所有的.jar和.zip文件
  4. File[] jarFiles = libPath.listFiles(new FilenameFilter() {
  5. public boolean accept(File dir, String name) {
  6. return name.endsWith(".jar") || name.endsWith(".zip");
  7. }
  8. });
  9. if (jarFiles != null) {
  10. // 从URLClassLoader类中获取类所在文件夹的方法
  11. // 对于jar文件,可以理解为一个存放class文件的文件夹
  12. Method method = URLClassLoader.class.getDeclaredMethod("addURL", URL.class);
  13. boolean accessible = method.isAccessible();     // 获取方法的访问权限
  14. try {
  15. if (accessible == false) {
  16. method.setAccessible(true);     // 设置方法的访问权限
  17. }
  18. // 获取系统类加载器
  19. URLClassLoader classLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
  20. for (File file : jarFiles) {
  21. URL url = file.toURI().toURL();
  22. try {
  23. method.invoke(classLoader, url);
  24. LOG.debug("读取jar文件[name={}]", file.getName());
  25. } catch (Exception e) {
  26. LOG.error("读取jar文件[name={}]失败", file.getName());
  27. }
  28. }
  29. } finally {
  30. method.setAccessible(accessible);
  31. }
  32. }

动态加载class文件

  1. // 设置class文件所在根路径
  2. // 例如/usr/java/classes下有一个test.App类,则/usr/java/classes即这个类的根路径,而.class文件的实际位置是/usr/java/classes/test/App.class
  3. File clazzPath = new File(class文件所在根路径);
  4. // 记录加载.class文件的数量
  5. int clazzCount = 0;
  6. if (clazzPath.exists() && clazzPath.isDirectory()) {
  7. // 获取路径长度
  8. int clazzPathLen = clazzPath.getAbsolutePath().length() + 1;
  9. Stack<File> stack = new Stack<>();
  10. stack.push(clazzPath);
  11. // 遍历类路径
  12. while (stack.isEmpty() == false) {
  13. File path = stack.pop();
  14. File[] classFiles = path.listFiles(new FileFilter() {
  15. public boolean accept(File pathname) {
  16. return pathname.isDirectory() || pathname.getName().endsWith(".class");
  17. }
  18. });
  19. for (File subFile : classFiles) {
  20. if (subFile.isDirectory()) {
  21. stack.push(subFile);
  22. } else {
  23. if (clazzCount++ == 0) {
  24. Method method = URLClassLoader.class.getDeclaredMethod("addURL", URL.class);
  25. boolean accessible = method.isAccessible();
  26. try {
  27. if (accessible == false) {
  28. method.setAccessible(true);
  29. }
  30. // 设置类加载器
  31. URLClassLoader classLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
  32. // 将当前类路径加入到类加载器中
  33. method.invoke(classLoader, clazzPath.toURI().toURL());
  34. } finally {
  35. method.setAccessible(accessible);
  36. }
  37. }
  38. // 文件名称
  39. String className = subFile.getAbsolutePath();
  40. className = className.substring(clazzPathLen, className.length() - 6);
  41. className = className.replace(File.separatorChar, '.');
  42. // 加载Class类
  43. Class.forName(className);
  44. LOG.debug("读取应用程序类文件[class={}]", className);
  45. }
  46. }
  47. }
  48. }

  完成上述两步操作后,即可使用Class.forName来加载jar中或.class文件包含的Java类了。