分享

java程序中动态加载jar包并执行

 陈永正的图书馆 2017-08-11

参考了几篇文章

加载jar包的类URLClassLoader中使用的方法为addURL,但是这个方法是protected类型


分别用了两种方法

一、

参考 http://blog.csdn.net/ozwarld/article/details/7440937

反射到这个类

  1. Method add = URLClassLoader.class.getDeclaredMethod("addURL", new Class[] { URL.class });    
  2. add.setAccessible(true);    

然后再执行add加载jar包,之后再次反射获取需要执行的类

  1. URLClassLoader classloader = (URLClassLoader)ClassLoader.getSystemClassLoader();    
  2. URL url = new File(jarPath).toURI().toURL();  
  3. add.invoke(classloader, new Object[] { url });   
  4. Class<?> clazz = Class.forName(className);  
  5. Method method = clazz.getDeclaredMethod("xxxx");  
  6. method.invoke(clazz.newInstance());  

最开始使用的是下面的方法,感觉简单,但是由于没设置parentClassLoader,导致出现各种bug,最后虽然调通了,但是依然选择了上面的方法(不确定是否还有其他问题)


二、

参考的http://www.cnblogs.com/flyingzl/articles/3139028.html

文章中先建立个子类,这样可以执行addURL这个方法了

  1. static class MyClassLoader extends URLClassLoader {  
  2.   
  3.         public MyClassLoader(URL[] urls) {  
  4.             super(urls);  
  5.         }  
  6.   
  7.         public MyClassLoader(URL[] urls, ClassLoader parent) {  
  8.             super(urls, parent);  
  9.         }  
  10.   
  11.         public void addJar(URL url) {  
  12.             this.addURL(url);  
  13.         }  
  14.   
  15.     }  

加载jar,并执行

  1. URL[] urls = new URL[] {};  
  2.        MyClassLoader classLoader = new MyClassLoader(urls,ClassLoader.getSystemClassLoader())  
  3.        try {  
  4.            classLoader.addJar(new File("c:/hello.jar").toURI().toURL());  
  5.            Class<?> clazz = classLoader.loadClass("pakcagename.classname");  
  6.            Method method = clazz.getDeclaredMethod("run");  
  7.            method.invoke(clazz.newInstance());  
  8.            classLoader.close();  
  9.        } catch (Exception e) {  
  10.            e.printStackTrace();  
  11.        }  


另、

如果是本地jar的话,这样就ok了,在hdfs上的话,会有点问题,URL不支持hdfs协议

  1. URL.setURLStreamHandlerFactory(new FsUrlStreamHandlerFactory());  

这样就支持hdfs协议了,path就可以转url了,当然,得加载下hdfs相关配置文件,获取FileSystem实例

  1. Path path = fs.getFileStatus(new Path(jarPath)).getPath();  
  2. URL url = path.toUri().toURL();  
  3. classLoader.addJar(url);  

只运行普通jar的话,在hdfs上是可以的


但是

如果加载的是mapreduce程序的jar包,只能是本地路径,不能使用hdfs路径,在setJarByClass方法中,是需要寻找到jar包绝对路径的


----------------------------------------------

使用一段时间后发现,进程是不停的,需要加载的jar包是可能需要修改的,所以导致多次加载不同版本的jar包之后,程序会有bug,读不到最新的jar包,不太可能更新jar包就重启下程序

于是,又研究了下classloarder,在当前线程加载jar包比较保险,线程结束会释放掉,保证每次加载都是最新的,而且也不占内存

  1. (URLClassLoader)ClassLoader.getSystemClassLoader();    
改为当前线程类加载器

  1. (URLClassLoader)Thread.currentThread().getContextClassLoader();    




    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多