关于ThreadContextLoader两行代码分析

1. 代码

 Class.forName("com.mysql.jdbc.Driver");
 Connection connection = DriverManager.getConnection("jdbc:mysql://localhost/liusheng", "xxx", "xx");

2. 第一行

我们都知道是向DriverManager注册驱动
因为 Class.forName 默认初始化类 ,所以以下代码会执行

 static {
        try {
            java.sql.DriverManager.registerDriver(new Driver());
        } catch (SQLException E) {
            throw new RuntimeException("Can't register driver!");
        }
    }

具体的是:

// 就是想CopyOnWriteList添加一个驱动(如果不存在)
registeredDrivers.addIfAbsent(new DriverInfo(driver, da));

3 第二行

就是获取连接 Connection 具体是
获取调用者的Class就是我们Main类

(getConnection(url, info, Reflection.getCallerClass()))

接下来:

// 获取我们调用者ClassLoader  (一般是Launcher$AppClassLoader)
    ClassLoader callerCL = caller != null ? caller.getClassLoader() : null;
        synchronized(DriverManager.class) {
        // 如果ClassLoader是Boot加载器 ,那么就设置为上下文加载器
            if (callerCL == null) {
                callerCL = Thread.currentThread().getContextClassLoader();
            }
        }
        SQLException reason = null;
       // 遍历说有注册的驱动
        for(DriverInfo aDriver : registeredDrivers) {
            if(isDriverAllowed(aDriver.driver, callerCL)) {
                try {
                   // 连接失败,或者 url 与驱动不匹配,那么久会抛异常,或者返回null ,由于下面会捕获异常,所以会继续尝试
                    Connection con = aDriver.driver.connect(url, info);
                    if (con != null) {
                        return (con);
                    }
                } catch (SQLException ex) {
                    if (reason == null) {
                        reason = ex;
                    }
                }

            } 

        }

        if (reason != null)    {
            println("getConnection failed: " + reason);
            throw reason;
        }
        throw new SQLException("No suitable driver found for "+ url, "08001");

判断方法 isDriverAllowed

private static boolean isDriverAllowed(Driver driver, ClassLoader classLoader) {
        boolean result = false;
        if(driver != null) {
            Class<?> aClass = null;
            try {
              // 回去调用者(对我们来说Main类)ClassLoader来加载
                aClass =  Class.forName(driver.getClass().getName(), true, classLoader);
            } catch (Exception ex) {
                result = false;
            }
            // 如果 两个Class是相同的,则表示驱动加载正确 ,这是为了防止,注册的实例和SPI(Service Provider Interface)使用的Class不同,从而出现错误,所以我们接下来让两个驱动不同,就会导致Class.forName 之后,也获取不到驱动
             result = ( aClass == driver.getClass() ) ? true : false;
        }

        return result;
    }

4.例子

1. 自定义ClassLoader

这个ClassLoader 从我们桌面获取Mysql驱动jar包,并加载

// 直接继承 URLClassLoader即可
public class MysqlClassLoade extends URLClassLoader {
    public MysqlClassLoade() throws MalformedURLException {
        super(new URL[]{Paths.get("C:\\Users\\Administrator\\Desktop\\mysql.jar").toUri().toURL()});
    }
}

2. Main类

public static void main(String[] args) throws ClassNotFoundException, MalformedURLException, SQLException {
       try {
           Class.forName("com.mysql.jdbc.Driver", true, new MysqlClassLoade());
       }catch (Exception e){
           System.out.println("加载 驱动失败");
       }
       try {
           Connection connection = DriverManager.getConnection("jdbc:mysql://134.175.9.88:3306/liusheng", "root", "root");
       }catch (Exception e){
           System.out.println("获取Connection 失败");
       }
   }

3 输出

image.png

说明驱动加载到了 ,但是连接没有获取到(注意:我们的Class Path 下不能有Mysql驱动,因为ClassLoader是双亲委托机制的 ,会先让父加载器加载一下)

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 孩子是父母爱情的结晶,孩子是上天赐予一个家庭里最好的礼物,每一对父母都希望自己的孩子在人生路上可以一帆风顺,顺风顺...
    红糖妈妈阅读 593评论 0 0
  • 我记得你望向我的样子 你记得我的样子 我们就全然爱过了
    寻桨阅读 420评论 5 8
  • 本文仅供学习,不构成买卖建议,据此买卖,后果自负,虚拟币交易风险大,投资需谨慎。 ================...
    水木山风阅读 709评论 2 2
  • 2009-11-15 很久很久以前,在遥远的银河系。 我在狠小的时候听过这首歌,不过只知道是《神雕侠侣》的插曲。 ...
    韩日记阅读 553评论 0 2
  • 战争,才是这个世界本来的面目。 任何一个文明的崛起,都伴随着战争。即便不是自己选的。立国70载的新中国,也是踏着朝...
    lysen963阅读 114评论 0 0

友情链接更多精彩内容