Service Provider Interface(SPI)是一种服务动态扩展机制,它提供接口对某种服务的行为进行定义和约束,第三方根据接口提供对该服务的一个具体实现,这些第三方的实现遵循同样的约束,相互可以替换,这允许服务的调用者根据场景选择不同的第三方实现,从而在运行时可以动态的扩展服务功能。
一个完整的SPI的框架应该包括下面几个关键组件:
1. 服务接口定义:包括一个或者多个接口/抽象类。
2. 服务提供者:服务的具体实现,通常以二进制库的方式提供。
3. 服务注册者:显示或者隐式的向系统注册一个或多个服务提供者。
Sun在Java类库开发中使用了不少基于SPI设计的库,比如JDBC,JCE,JNDI,JAXP,JBI,NIO, Java Sound, Java Image I/O, Java Locale(java.util.spi)等。以JDBC为例,java.sql.Driver是服务接口定义,各种jdbc的驱动是服务的提供者,比如Oracle的thin driver, mysql的driver,这些服务提供者以jar包的形式提供给服务使用者,在使用前需要显示的注册到系统里, java.sql.DriverManager充当了服务注册者的角色。
下面的代码注册mysql driver并创建一个java.sql.Connection.
Class.forName(“com.mysql.jdbc.Driver”);
String url = "jdbc:mysql://127.0.0.1/test?useUnicode=true&characterEncoding=utf-8";
String user = "";
String passwd = "";
Connection conn = DriverManager.getConnection(url, user, passwd);
当“com.mysql.jdbc.Driver”被加载的时候,它通过一个static块创建一个实例, 并调用DriverManager的registerDriver()方法把自己注册到DriverManager,当调用getConnection()的时候,DriverManager会调用已经注册的mysql Driver来创建Connection实例。
JAVA类库中更普遍的是提供API(Application Programming Interface),例如集合API,并发框架的API等等。API和SPI面向的用户是不一样的,API面向应用的开发者,开发者直接调用API来实现某种功能,而SPI是面向第三方厂商,第三方厂商实现SPI并以插件的方式提供某种功能。对于应用的开发者来说,SPI是透明的,开发者只需要关心如何使用API。以上面JDBC为例,java.sql.Driver是一个SPI,但是编写代码的时候是使用Connection这个API。