为什么需要提供者接口?难道我们不能自己轻松地注册服务吗?例如,维护服务实现的映射,然后在查找时返回实例?
正如其他人所说,Provider 的目的是拥有一个可以创建Service
实例的 AbstractFactory。您并不总是希望保留对所有服务实现的引用,因为它们可能是短暂的和/或在执行后可能无法重用。
但是提供者的目的是什么?如果没有提供者,如何使用“提供者注册 API”?
拥有 Provider 接口的最有力的原因之一是您不需要在编译时实现。API 的用户可以稍后添加他们自己的实现。
让我们以 JDBC 为例,就像在另一个答案中使用的 Ajay 一样,但让我们更进一步:
有许多不同类型的数据库和数据库供应商,它们管理和实现数据库的方式(以及查询它们的方式)略有不同。Java 的创建者不可能创建所有这些不同可能方式的实现,原因有很多:
- 最初编写 Java 时,许多这样的数据库公司或系统还不存在。
- 并非所有这些数据库供应商都是开源的,因此 Java 的创建者不知道如何与他们沟通,即使他们愿意。
- 用户可能想要编写自己的自定义数据库
那么你如何解决这个问题呢?通过使用Service Provider
.
- 驱动程序
Provider
接口是. 它提供了与特定供应商的数据库交互的方法。其中一种方法Driver
是工厂方法,用于在给定 url 和其他属性(如用户名和密码等)的情况下为数据库创建Connection
实例(即Service
)。
每个数据库供应商都编写了自己的Driver
实现,以了解如何与自己的数据库系统进行通信。这些不包含在 JDK 中;您必须访问公司网站或其他代码存储库并将它们作为单独的 jar 下载。
要使用这些驱动程序,您必须将 jar 添加到类路径中,然后使用 JDKDriverManager
类来注册驱动程序。
DriverManager 类有一个方法registerDriver(Driver)
,用于在服务注册中注册一个 Driver 实例,以便可以使用它。按照惯例,大多数Driver
实现在类加载时注册,因此您在代码中所要做的就是编写
Class.forname("foo.bar.Driver");
为供应商“foo.bar”注册驱动程序(假设您的类路径中有该类的 jar。)
注册数据库驱动程序后,您可以获得连接到数据库的服务实现实例。
例如,如果您在本地计算机上有一个名为“test”的 mysql 数据库,并且您的用户帐户的用户名为“monty”,密码为“greatsqldb”,那么您可以像这样创建一个服务实现:
Connection conn =
DriverManager.getConnection("jdbc:mysql://localhost/test?" +
"user=monty&password=greatsqldb");
DriverManager 类看到你传入的 String 并找到可以理解其含义的注册驱动程序。(这实际上是Chain of Responsibility
通过遍历所有注册的驱动程序并调用它们的Driver.acceptsUrl(Stirng)
方法直到 url 被接受来使用 Pattern 完成的)
请注意,JDK 中没有特定于 mysql 的代码。您所要做的就是注册某个供应商的驱动程序,然后将格式正确的字符串传递给服务提供者。如果我们稍后决定使用不同的数据库供应商(如 oracle 或 sybase),那么我们只需交换 jar 并修改我们的连接字符串。DriverManager 中的代码不会改变。
为什么我们不只是建立一次联系并保持联系呢?为什么我们需要服务?
我们可能希望在每次操作后连接/断开连接。或者我们可能希望将连接保持更长时间。拥有该服务允许我们随时创建新的连接,并且不妨碍我们保留对它的引用以供以后重复使用。
这是一个非常强大的概念,框架使用它来允许许多可能的排列和扩展,而不会弄乱核心代码库。
编辑
使用多个提供者和提供多个的提供者Services
:
没有什么可以阻止您拥有多个提供者。您可以同时连接到使用不同数据库供应商软件创建的多个数据库。您还可以同时连接到同一供应商生产的多个数据库。
多种服务 - 一些提供者甚至可能Service
根据连接 url 提供不同的实现。例如,H2 可以创建基于文件系统或基于内存的数据库。告诉 H2 你想使用哪个的方法是不同的 url 格式。我没有看过 H2 代码,但我假设基于文件和基于内存是不同的服务实现。
为什么 DriverManager 不只管理 Connections 而 Oracle 可以实现 OracleConnectionWrapper?没有供应商!
这还需要您知道您有一个 Oracle 连接。那是非常紧密的耦合,如果我改变供应商,我将不得不改变很多代码。
Service Registration
只需要一个字符串。请记住,它使用chain of Responsiblity
来查找第一个知道如何处理 url 的注册提供程序。应用程序可以是供应商中立的,它可以从属性文件中获取连接 url 和驱动程序类名称。这样,如果我更改供应商,我就不必重新编译我的代码。但是,如果我硬编码对“OracleConnectionWrapper”的引用,然后我更改了供应商,我将不得不重写部分代码,然后重新编译。
如果他们愿意,没有什么可以阻止某人支持多种数据库供应商 url 格式。因此,如果我愿意,我可以制作一个可以处理 mysql 和 oracle 的 GenericDriver。