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

参考了几篇文章

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

分别用了两种方法

一、

反射到这个类

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

add.setAccessible(true);

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

URLClassLoader classloader = (URLClassLoader)ClassLoader.getSystemClassLoader();

URL url = new File(jarPath).toURI().toURL();

add.invoke(classloader, new Object[] { url });

Class> clazz = Class.forName(className);

Method method = clazz.getDeclaredMethod("xxxx");

method.invoke(clazz.newInstance());

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

二、

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

static class MyClassLoader extends URLClassLoader {

public MyClassLoader(URL[] urls) {

super(urls);

}

public MyClassLoader(URL[] urls, ClassLoader parent) {

super(urls, parent);

}

public void addJar(URL url) {

this.addURL(url);

}

}

加载jar,并执行

URL[] urls = new URL[] {};

MyClassLoader classLoader = new MyClassLoader(urls,ClassLoader.getSystemClassLoader())

try {

classLoader.addJar(new File("c:/hello.jar").toURI().toURL());

Class> clazz = classLoader.loadClass("pakcagename.classname");

Method method = clazz.getDeclaredMethod("run");

method.invoke(clazz.newInstance());

classLoader.close();

} catch (Exception e) {

e.printStackTrace();

}

另、

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

URL.setURLStreamHandlerFactory(new FsUrlStreamHandlerFactory());

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

Path path = fs.getFileStatus(new Path(jarPath)).getPath();

URL url = path.toUri().toURL();

classLoader.addJar(url);

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

但是

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

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

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

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

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

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

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值