【JVM】JDBC案例打破双亲委派机制


🐌个人主页: 🐌 叶落闲庭

💨我的专栏:💨
c语言
数据结构
javaEE
操作系统
Redis

石可破也,而不可夺坚;丹可磨也,而不可夺赤。


JVM

打破双亲委派机制(JDBC案例)

JDBC的这个框架的目的是在Java中操作数据库,在他的设计的核心思想中不希望出现具体的数据库的语法,提高它的泛用性,将来对接任何数据库都会比较容易,在JDBC中,设计了一个DriverManager的类,这个类会去管理在jar包中的引入的数据库的驱动,比如要使用mysql数据库,那么就会添加一个mysql的驱动jar包,DriverManager会负责把驱动jar包加载进来,这样就可以连接mysql的数据库了,要想连接其他类型的数据库,只需将jar包替换成其他数据库的即可,这样DriverManager就可以对接不同的数据库,并且在代码方面是统一的,但是在DriverManager去加载驱动jar包的类的过程中打破了双亲委派机制:

DriverManager这个类是jdk自己提供的,位于rt.jar这个jar包中,所以这个类就会由启动类加载器加载,而mysql的驱动jar包是由应用程序类加载器加载的,当启动类加载器加载完DriverManager类后,又将DriverManager委派给应用程序类加载器去加载mysql的驱动jar包,这就违反了双亲委派机制



问题1:DriverManager怎么知道jar包中要加载的驱动在哪?

SPI机制

  • spi全称为Service Provider Interface ,是JDK内置的一种服务提供发现机制
  • spi工作原理:

以jdbc连接数据库为例:

  • 1.在ClassPath路径下的META-INF/services文件夹中,以接口的全限定名来命名文件名,对应的文件里写该接口的实现,这个文件的名字就是当前驱动实现的接口,DriverManager都是实现了这个接口的,才会认为它是一个数据库驱动,所以mysql的jar包的驱动也实现了这个接口,在文件的内部就需要写上需要暴露的接口的的实现类,比如mysql某个版本的驱动就是com.mysql.jdbc.Driver
  • 2.此时DriverManager就可以去拿到当前的实现类的类名,并且用类加载器加载,在spi机制中提供了一个类ServiceLoader,只需把想要的接口的名字传入即可,接下来会有一个加载器,这个加载器就可以使用迭代器拿到当前的类名并且创建这个对象
    ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);


依据代码调试演示DriverManager加载驱动jar包

1.基础JDBC代码

java 复制代码
public class JDBCExample {
    static final String JDBC_DROVER = "com.mysql.cj.jdbc.Driver";
    static final String DB_URL = "jdbc:mysql:///demo1";
    static final String USER = "root";
    static final String PASS = "123456";

    public static void main(String[] args) {
        Connection coon = null;
        Statement stmt = null;
        try {
            coon = DriverManager.getConnection(DB_URL,USER,PASS);
            stmt = coon.createStatement();
            String sql = "select name,author from book";
            ResultSet r = stmt.executeQuery(sql);
            while (r.next()) {
                System.out.print("name: " + r.getString("name"));
                System.out.print("author: " + r.getString("author") + "\n");
            }
            stmt.close();
            coon.close();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}

2. 初始化

java 复制代码
static {
    loadInitialDrivers();
    println("JDBC DriverManager initialized");
}

3. 加载所有jar包中的驱动

  • 核心代码:
java 复制代码
public Void run() {
     ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
     Iterator<Driver> driversIterator = loadedDrivers.iterator();
     try{
         	while(driversIterator.hasNext()) {
         		driversIterator.next();
         	}
       } catch(Throwable t) {
         // Do nothing
     }
    return null;
}



问题2:SPI中是如何获取到应用程序类加载器的?

  • SPI中使用了线程上下文中保存的类加载器进行类的加载,这个类加载器一般是应用程序类加载器。

获取当前类的上下文的类加载器



相关推荐
程序员卷卷狗1 天前
JVM 调优实战:从线上问题复盘到精细化内存治理
java·开发语言·jvm
Sincerelyplz2 天前
【JDK新特性】分代ZGC到底做了哪些优化?
java·jvm·后端
初学小白...2 天前
线程同步机制及三大不安全案例
java·开发语言·jvm
凤山老林3 天前
还在用JDK8?JDK8升级JDK11:一次价值千万的升级指南
java·开发语言·jvm·spring boot·后端·jdk
2501_938790073 天前
详解 JVM 中的对象创建过程:类加载检查、内存分配、初始化的完整流程
jvm
宸津-代码粉碎机3 天前
Java内部类内存泄露深度解析:原理、场景与根治方案(附GC引用链分析)
java·开发语言·jvm·人工智能·python
杨筱毅3 天前
【底层机制】Android GC -- 为什么要有GC?GC的核心原理?理解GC的意义
android·jvm·gc
東雪木3 天前
Java基础语言进阶学习——1,JVM内存模型(堆、栈、方法区)
java·jvm·学习
小满、3 天前
JVM 执行引擎:字节码是如何被执行的
jvm·字节码·jvm执行引擎
无敌最俊朗@3 天前
SQLite 约束 (Constraints) 面试核心知识点
java·开发语言·jvm