当前位置:   article > 正文

【Web】浅聊JDBC的SPI机制是怎么实现的——DriverManager

【Web】浅聊JDBC的SPI机制是怎么实现的——DriverManager

目录

前言

分析


前言

【Web】浅浅地聊JDBC java.sql.Driver的SPI后门-CSDN博客

上篇文章我们做到了知其然,知道了JDBC有SPI机制,并且可以利用其Driver后门

这篇文章希望可以做到知其所以然,对JDBC的SPI机制的来源做到心里有数

分析

先是回顾一下JDBC的代码

  1. package com.spi;
  2. import java.sql.Connection;
  3. import java.sql.DriverManager;
  4. import java.sql.ResultSet;
  5. import java.sql.Statement;
  6. public class Jdbc_Demo {
  7. public static void main(String[] args) throws Exception {
  8. // 注册驱动(可以删去)
  9. // Class.forName("java.sql.Driver");
  10. // 获取数据库连接对象
  11. Connection con = DriverManager.getConnection("jdbc:mysql://localhost:3306/security","root", "root");
  12. // 定义sql语句
  13. String sql = "select * from users";
  14. // 获取执行sql的对象Statement
  15. Statement stmt = con.createStatement();
  16. // 执行sql
  17. ResultSet res = stmt.executeQuery(sql);
  18. // 处理对象
  19. while(res.next()) {
  20. System.out.println(res.getString("username"));
  21. }
  22. // 释放资源
  23. res.close();
  24. stmt.close();
  25. con.close();
  26. }
  27. }

我们可以看到

 Connection con = DriverManager.getConnection("jdbc:mysql://localhost:3306/security","root", "root");

因此JVM会尝试去加载DriverManager类,进而执行DriverManager的静态代码,调用类中的loadInitialDrivers方法(稍微精简了一下代码)

  1. static {
  2. loadInitialDrivers();
  3. println("JDBC DriverManager initialized");
  4. }
  5. private static void loadInitialDrivers() {
  6. String drivers;
  7. try {
  8. drivers = AccessController.doPrivileged(new PrivilegedAction<String>() {
  9. public String run() {
  10. return System.getProperty("jdbc.drivers"); // 也可以通过设置系统属性来加载驱动
  11. }
  12. });
  13. } //...
  14. AccessController.doPrivileged(new PrivilegedAction<Void>() {
  15. public Void run() {
  16. ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
  17. Iterator<Driver> driversIterator = loadedDrivers.iterator();
  18. try{
  19. while(driversIterator.hasNext()) {
  20. driversIterator.next();
  21. }
  22. }
  23. // ....
  24. String[] driversList = drivers.split(":");
  25. for (String aDriver : driversList) {
  26. try {
  27. Class.forName(aDriver, true,
  28. ClassLoader.getSystemClassLoader());
  29. } // ...
  30. }
  31. }

 重点关注ServiceLoader.load(Driver.class)进行了类加载

public static <S> ServiceLoader<S> load(Class<S> service) {
    ClassLoader cl = Thread.currentThread().getContextClassLoader();
    return ServiceLoader.load(service, cl);
}

 跟进ServiceLoader.load(service, cl);

public static <S> ServiceLoader<S> load(Class<S> service,
                                            ClassLoader loader)
    {
        return new ServiceLoader<>(service, loader);
    }

service是Driver.class,loader是 Thread.currentThread().getContextClassLoader()

这里其实还有双亲委派机制的打破可以讲,但强行塞有点过于臃肿了,略去

 跟进ServiceLoader<>(service, loader);

private ServiceLoader(Class<S> svc, ClassLoader cl) {
    service = Objects.requireNonNull(svc, "Service interface cannot be null");
    loader = (cl == null) ? ClassLoader.getSystemClassLoader() : cl;
    acc = (System.getSecurityManager() != null) ? AccessController.getContext() : null;
    reload();
}

跟进reload();

public void reload() {
    providers.clear();
    lookupIterator = new LazyIterator(service, loader);
}

最后得到的driversIterator就是LazyIterator

OK我们再看下上面代码的这段局部

  1. while(driversIterator.hasNext()) {
  2. driversIterator.next();
  3. }

在这段代码中调用 driversIterator.next()之时,便是调用LazyIterator#next

  1. public S next() {
  2. if (acc == null) {
  3. return nextService();
  4. } else {
  5. PrivilegedAction<S> action = new PrivilegedAction<S>() {
  6. public S run() { return nextService(); }
  7. };
  8. return AccessController.doPrivileged(action, acc);
  9. }
  10. }

跟进LazyIterator#nextService

  1. private S nextService() {
  2. if (!hasNextService())
  3. throw new NoSuchElementException();
  4. String cn = nextName;
  5. nextName = null;
  6. Class<?> c = null;
  7. try {
  8. c = Class.forName(cn, false, loader);
  9. }

这段代码的作用是根据存储的类名动态加载一个类(Class.forName)

要注意一点:将 initialize 参数设置为 false,可以实现延迟类的静态初始化,静态初始化块和静态变量的赋值是在类第一次被加载时执行的,如果将 initialize 参数设置为 false,则这些静态操作将被延迟到后续使用该类的过程中才被执行

类名是哪来的?张口就来吗?我知道你很急,但你先别急,下面就讲。

 String cn = nextName;

看到nextName直接就懂了哇,回头看driversIterator.hasNext(),即LazyIterator#hasNext

  1. public boolean hasNext() {
  2. if (acc == null) {
  3. return hasNextService();
  4. } else {
  5. PrivilegedAction<Boolean> action = new PrivilegedAction<Boolean>() {
  6. public Boolean run() { return hasNextService(); }
  7. };
  8. return AccessController.doPrivileged(action, acc);
  9. }
  10. }

跟进LazyIterator#hasNextService

  1. private boolean hasNextService() {
  2. if (nextName != null) {
  3. return true;
  4. }
  5. if (configs == null) {
  6. try {
  7. String fullName = PREFIX + service.getName();
  8. if (loader == null)
  9. configs = ClassLoader.getSystemResources(fullName);
  10. else
  11. configs = loader.getResources(fullName);
  12. } catch (IOException x) {
  13. fail(service, "Error locating configuration files", x);
  14. }
  15. }
  16. while ((pending == null) || !pending.hasNext()) {
  17. if (!configs.hasMoreElements()) {
  18. return false;
  19. }
  20. pending = parse(service, configs.nextElement());
  21. }
  22. nextName = pending.next();
  23. return true;
  24. }

ServiceLoader<S>类有常量属性PREFIX = "META-INF/services/" 

service是传入的Driver.class,service.getName()获取Driver接口全类名,拼接得到SPI文件名,给后续读取得到实现类的类名,最后赋值给nextName,再交给LazyIterator#nextService去进行类的动态加载

  1. while ((pending == null) || !pending.hasNext()) {
  2. if (!configs.hasMoreElements()) {
  3. return false;
  4. }
  5. pending = parse(service, configs.nextElement());
  6. }
  7. nextName = pending.next();
  8. return true;

上面这段话,白话来讲就是,JVM去META-INF/services/下去找Driver接口名的文件,把文件中的内容读出来,也就是我们所要加载的类名,并交由Class.forName来进行动态类的加载。

至此,SPI大成!

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/从前慢现在也慢/article/detail/204640
推荐阅读
相关标签
  

闽ICP备14008679号