天天看点

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);
           
  • 小结

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

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

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