我懵了,那个听起来很厉害的微内核架构是个什么鬼?

private final AtomicLong atomicId = new AtomicLong(100L);

@Override

public String generateId() {

long leastId = this.atomicId.incrementAndGet();

return String.valueOf(leastId);

}

}

  • 在项目的resources/META-INF/services 目录下添加一个名为com.github.jianzh5.spi.IdGenerator的文件,这是 JDK SPI 需要读取的配置文件,内容如下:

com.github.jianzh5.spi.impl.UuidGenerator

com.github.jianzh5.spi.impl.SequenceIdGenerator

  • 创建main方法,让其加载上述的配置文件,创建全部IdGenerator 接口实现的实例,并执行生成id的方法。

public class GeneratorMain {

public static void main(String[] args) {

ServiceLoader serviceLoader = ServiceLoader.load(IdGenerator.class);

Iterator iterator = serviceLoader.iterator();

while(iterator.hasNext()){

IdGenerator generator = iterator.next();

String id = generator.generateId();

System.out.println(generator.getClass().getName() + "  >>id:" + id);

}

}

}

  • 执行结果如下:

6541e9ee312a29a08897dadde22418c6.png

JDK SPI 源码分析

通过上述示例,我们可以看到 JDK SPI 的入口方法是 ServiceLoader.load() 方法,在这个方法中首先会尝试获取当前使用的 ClassLoader,然后调用 reload() 方法,调用关系如下图所示:

03d8d2209812fabb55025ddc8e67deca.png

public void reload() {

providers.clear();

lookupIterator = new LazyIterator(service, loader);

}

在前面的示例中,main() 方法中使用的迭代器底层就是调用了 ServiceLoader.LazyIterator 实现的。Iterator 接口有两个关键方法:hasNext() 方法和 next() 方法。这里的 LazyIterator 中的 next() 方法最终调用的是其 nextService() 方法,hasNext() 方法最终调用的是 hasNextService() 方法,我们来看看 hasNextService()方法的具体实现:

private static final String PREFIX = “META-INF/services/”;

Enumeration configs = null;

Iterator pending = null;

String nextName = null;

private boolean hasNextService() {

if (nextName != null) {

return true;

}

if (configs == null) {

try {

//META-INF/services/com.github.jianzh5.spi.IdGenerator

String fullName = PREFIX + service.getName();

if (loader == null)

configs = ClassLoader.getSystemResources(fullName);

else

configs = loader.getResources(fullName);

} catch (IOException x) {

fail(service, “Error locating configuration files”, x);

}

}

// 按行SPI遍历配置文件的内容

while ((pending == null) || !pending.hasNext()) {

if (!configs.hasMoreElements()) {

return false;

}

// 解析配置文件

pending = parse(service, configs.nextElement());

}

// 更新 nextName字段

nextName = pending.next();

return true;

}

在 hasNextService() 方法中完成 SPI 配置文件的解析之后,再来看 LazyIterator.nextService() 方法,该方法负责实例化 hasNextService() 方法读取到的实现类,其中会将实例化的对象放到 providers 集合中缓存起来,核心实现如下所示:

private S nextService() {

String cn = nextName;

nextName = null;

// 加载 nextName字段指定的类

Class<?> c = Class.forName(cn, false, loader);

if (!service.isAssignableFrom©) { // 检测类型

fail(service, “Provider " + cn  + " not a subtype”);

}

S p = service.cast(c.newInstance()); // 创建实现类的对象

providers.put(cn, p); // 将实现类名称以及相应实例对象添加到缓存

return p;

}

以上就是在 main() 方法中使用的迭代器的底层实现。最后,我们再来看一下 main() 方法中使用 ServiceLoader.iterator() 方法拿到的迭代器是如何实现的,这个迭代器是依赖 LazyIterator 实现的一个匿名内部类,核心实现如下:

public Iterator iterator() {

return new Iterator() {

// knownProviders用来迭代providers缓存

Iterator<Map.Entry<String,S>> knownProviders

= providers.entrySet().iterator();

public boolean hasNext() {

// 先走查询缓存,缓存查询失败,再通过LazyIterator加载

if (knownProviders.hasNext())

return true;

return lookupIterator.hasNext();

}

public S next() {

// 先走查询缓存,缓存查询失败,再通过 LazyIterator加载

if (knownProviders.hasNext())

return knownProviders.next().getValue();

return lookupIterator.next();

}

// 省略remove()方法

};

}

JDK SPI 在 JDBC 中的应用

了解了 JDK SPI 实现的原理之后,我们再来看实践中 JDBC 是如何使用 JDK SPI 机制加载不同数据库厂商的实现类。

JDK 中只定义了一个 java.sql.Driver 接口,具体的实现是由不同数据库厂商来提供的。这里我们就以 MySQL 提供的 JDBC 实现包为例进行分析。

在 mysql-connector-java-*.jar 包中的 META-INF/services 目录下,有一个 java.sql.Driver 文件中只有一行内容,如下所示:

com.mysql.cj.jdbc.Driver

在使用 mysql-connector-java-*.jar 包连接 MySQL 数据库的时候,我们会用到如下语句创建数据库连接:

String url = “jdbc:xxx://xxx:xxx/xxx”;

Connection conn = DriverManager.getConnection(url, username, pwd);

DriverManager 是 JDK 提供的数据库驱动管理器,其中的代码片段,如下所示:

static {

loadInitialDrivers();

println(“JDBC DriverManager initialized”);

}

在调用 getConnection() 方法的时候,DriverManager 类会被 Java 虚拟机加载、解析并触发 static 代码块的执行;在 loadInitialDrivers()方法中通过 JDK SPI 扫描 Classpath 下 java.sql.Driver 接口实现类并实例化,核心实现如下所示:

private static void loadInitialDrivers() {

String drivers = System.getProperty(“jdbc.drivers”)

// 使用 JDK SPI机制加载所有 java.sql.Driver实现类

ServiceLoader loadedDrivers =

ServiceLoader.load(Driver.class);

Iterator driversIterator = loadedDrivers.iterator();

while(driversIterator.hasNext()) {

driversIterator.next();

}

String[] driversList = drivers.split(“:”);

for (String aDriver : driversList) { // 初始化Driver实现类

Class.forName(aDriver, true,

ClassLoader.getSystemClassLoader());

}

}

在 MySQL 提供的 com.mysql.cj.jdbc.Driver 实现类中,同样有一段 static 静态代码块,这段代码会创建一个 com.mysql.cj.jdbc.Driver 对象并注册到 DriverManager.registeredDrivers 集合中(CopyOnWriteArrayList 类型),如下所示:

static {

java.sql.DriverManager.registerDriver(new Driver());

}

在 getConnection() 方法中,DriverManager 从该 registeredDrivers 集合中获取对应的 Driver 对象创建 Connection,核心实现如下所示:

private static Connection getConnection(String url, java.util.Properties info, Class<?> caller) throws SQLException {

// 省略 try/catch代码块以及权限处理逻辑

for(DriverInfo aDriver : registeredDrivers) {

最后:学习总结——MyBtis知识脑图(纯手绘xmind文档)

学完之后,若是想验收效果如何,其实最好的方法就是可自己去总结一下。比如我就会在学习完一个东西之后自己去手绘一份xmind文件的知识梳理大纲脑图,这样也可方便后续的复习,且都是自己的理解,相信随便瞟几眼就能迅速过完整个知识,脑补回来。下方即为我手绘的MyBtis知识脑图,由于是xmind文件,不好上传,所以小编将其以图片形式导出来传在此处,细节方面不是特别清晰。但可给感兴趣的朋友提供完整的MyBtis知识脑图原件(包括上方的面试解析xmind文档)

image

除此之外,前文所提及的Alibaba珍藏版mybatis手写文档以及一本小小的MyBatis源码分析文档——《MyBatis源码分析》等等相关的学习笔记文档,也皆可分享给认可的朋友!

学习总结——MyBtis知识脑图(纯手绘xmind文档)

学完之后,若是想验收效果如何,其实最好的方法就是可自己去总结一下。比如我就会在学习完一个东西之后自己去手绘一份xmind文件的知识梳理大纲脑图,这样也可方便后续的复习,且都是自己的理解,相信随便瞟几眼就能迅速过完整个知识,脑补回来。下方即为我手绘的MyBtis知识脑图,由于是xmind文件,不好上传,所以小编将其以图片形式导出来传在此处,细节方面不是特别清晰。但可给感兴趣的朋友提供完整的MyBtis知识脑图原件(包括上方的面试解析xmind文档)

[外链图片转存中…(img-fjNwjMLC-1720108547355)]

除此之外,前文所提及的Alibaba珍藏版mybatis手写文档以及一本小小的MyBatis源码分析文档——《MyBatis源码分析》等等相关的学习笔记文档,也皆可分享给认可的朋友!

  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值