java中的SPI机制
1.手动注册驱动
记得以前写连接数据库代码的时候大概分了这么几部:
1)注册驱动;
2)获取数据库连接;
3)编写sql;
4)执行sql;
5)遍历结果集
在注册驱动的时候会有这样的代码
Class.forName("com.mysql.jdbc.Driver")
因为数据库的种类有很多,所以java提供一个jdbc的接口,来让不同的数据库厂商去实现。这些不同的驱动其实就是不同数据库厂商提供的连接他们数据库的实现类。在调用mysql的驱动类的时候要保证这些类已经加载过了,调用Class.forName("com.mysql.jdbc.Driver")
过程中会在加载类的时候向DriverManger注册自己,mysql类中的静态代码块为:
package com.mysql.cj.jdbc;
import java.sql.SQLException;
public class Driver extends NonRegisteringDriver implements java.sql.Driver {
//
// Register ourselves with the DriverManager
//
static {
try {
//向DriverManager注册自己
java.sql.DriverManager.registerDriver(new Driver());
} catch (SQLException E) {
throw new RuntimeException("Can't register driver!");
}
}
/**
* Construct a new driver and register it with DriverManager
*
* @throws SQLException
* if a database error occurs.
*/
public Driver() throws SQLException {
// Required for Class.forName().newInstance()
}
}
静态代码块中的代码会在加载类的时候执行一次,所以可以正常的获取到数据库的连接。
而在jdbc4.0之后,支持了SPI机制,就不需要手动注册驱动类了。
2.)spi机制
spi(Service Provider Interface),是JDK内置的服务提供机制。通过在jar包下的META-INF/services/xxx(你要实现的接口)
文件中声明该接口的实现类,就可以通过ServiceLoader.load(xxx.class)
扫描到然后加载该类,实现类为在该jar包下的全限定名。例如mysql的jar包中
mysql的spi的关键代码
//DriverManger中静态代码块为
static {
loadInitialDrivers();
println("JDBC DriverManager initialized");
}
通过serviceLoader.load(Driver,class)获得实现类,然后迭代遍历所有的实现类。
参考:
[aoho]:https://juejin.cn/post/6844903605695152142#comment