ClassLoader 解决问题的一次小思考

ClassLoader的一次小理解

背景

今天群里一哥们说了一个问题,试了好几下才解决。有点惭愧,记录之。

需求:
业务方要求,需要访问sqlite数据库,但是不能提前把sqlite的jar包放到工程里。( 可能是为了实现技术上的所谓的热拔插吧)

探索解决

需求是否合理,到了开发解决,只有使命必达。
当然,前提是有可行性,否则你可以拿着40米的大刀,让需求侧同学先跑39米 :))

热拔插,那就拿出反射来搞一把呗。
于是乎,有了一下过程

  • 第一版代码
String path ="file:///User/xx/tmp/sqlite-jdbc-3.8.7.jar";
            URL url = new URL(path);
            URLClassLoader myClassLoader = new URLClassLoader(new URL[] {url},Thread.currentThread()
                    .getContextClassLoader());
            Class<?> driverClass = myClassLoader.loadClass("org.sqlite.JDBC");
            Class.forName("org.sqlite.JDBC",true, myClassLoader);


            String dbURL="jdbc:sqlite:C:/Users/Administrator/Downloads/test.db";

            Class<?> dmClazz = myClassLoader.loadClass("java.sql.DriverManager");
            Method method = dmClazz.getMethod("getConnection", Connection.class);
            Connection conn  = (Connection)method.invoke(null,dbURL);

以上代码用了反射,类加载器,指导思想是,既然是动态加载,那就全部用新的classload 搞进来,不就可以了吗?

结果,不行。 因为DriverManager getConnection 是个静态方法,而且没有给你指定loader的地方。 即使自己load近来,依然找不到驱动。


  • 思考总结

上述代码,把已经想到的构造一个classloader 反射调用方法,获取Connection,都用了,发现还是不行。

此时,喝杯茶,静一下。

翻看一下代码

Class.forName();
DriverManager.getConnection

看了一下相关注释,JDBC本身就是用反射获取驱动类,进行管理。

所以这时,思考了一下,是否换个思路,把驱动jar包放到当前loader的classpath即可。


  • 解决方案

指导思想:
1.获取当前loader
2.获取相关jar包
3.放进当前loader对应的classpath

代码如下:

//1.获取jar
			String path ="file:///User/xx/tmp/sqlite-jdbc-3.8.7.jar";
            URL url = new URL(path);
            //2获取当前loader
            URLClassLoader classLoader = (URLClassLoader) ClassLoader.getSystemClassLoader();
			//3. 把外部临时加载的jar包放进classpath
            Method method = URLClassLoader.class.getDeclaredMethod("addURL", URL.class);
            boolean accessible=method.isAccessible();
            method.setAccessible(true);
            
            method.invoke(classLoader, url);
            
            method.setAccessible(accessible);

  • 小结

回顾解决问题的过程,其实刚开始受问题发起者的影响较大,绕进了技术范畴。

遇阻后,倒过来根据经验解决问题,反倒立马解决

很多时间,不是技术问题,是经验问题

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值