我正在阅读Bloch的有效Java书籍[1],并遇到了以下SPI示例:
//Service interface
public interface Service {
//Service specific methods here
}
//Service provider interface
public interface Provider {
Service newService();
}
//Class for service registration and access
public class Services {
private Services(){}
private static final Map<String, Provider> providers =
new ConcurrentHashMap<String, Provider>();
public static final String DEFAULT_PROVIDER_NAME = "<def>";
//Registration
public static void registerDefaultProvider(Provider p) {
registerProvider(DEFAULT_PROVIDER_NAME, p);
}
public static void registerProvider(String name, Provider p) {
providers.put(name, p);
}
//Access
public static Service newInstance() {
return newInstance(DEFAULT_PROVIDER_NAME);
}
public static Service newInstance(String name) {
// you get the point..lookup in the map the provider by name
// and return provider.newService();
}
最佳答案
为什么需要提供程序接口(interface)?我们无法像自己一样轻松地注册服务吗?维护服务实现的 map ,然后在查找时返回实例?
如其他人所述,Provider的目的是拥有一个可以制作Service实例的AbstractFactory。您并不总是希望保留对所有Service实现的引用,因为它们可能是短暂的和/或在执行后可能不可重用。
但是提供者的目的是什么,如果没有提供者,如何使用“提供者注册API”
拥有Provider接口(interface)的最强大的原因之一就是您不需要在编译时就实现。 API的用户可以在以后添加自己的实现。
让我们使用JDBC作为示例,就像在另一个答案中使用的Ajay一样,但让我们更进一步:
有许多不同类型的数据库和数据库供应商,它们的管理和实现数据库(以及如何查询数据库)的方式略有不同。 Java的创建者可能由于多种原因无法创建所有这些不同可能方式的实现:
Service Provider。Provider。它提供了与特定供应商的数据库进行交互的方法。 Driver中的一种方法是一种工厂方法,用于在给定URL和其他属性(例如用户名和密码等)的情况下,将Connection实例(即Service)制作到数据库中。 Driver实现,以实现如何与自己的数据库系统进行通信。这些未包含在JDK中;您必须访问公司网站或其他代码存储库,并将其作为单独的jar下载。DriverManager类注册该驱动程序。Service Registration。 registerDriver(Driver)方法,该方法用于在服务注册中注册驱动程序实例,以便可以使用它。按照惯例,大多数Driver实现都会在类加载时进行注册,因此您在代码中要做的就是编写Class.forname("foo.bar.Driver");
Connection conn =
DriverManager.getConnection("jdbc:mysql://localhost/test?" +
"user=monty&password=greatsqldb");
Chain of Responsibility模式完成的,方法是遍历所有已注册的驱动程序并调用其Driver.acceptsUrl(Stirng)方法,直到该URL被接受为止)Services的提供程序一起工作:Service实现。例如,H2可以创建基于文件系统或基于内存的数据库。告诉H2您要使用哪一个的方法是另一种url格式。我没有看过H2代码,但是我假设基于文件和基于内存的是不同的Service实现。Service Registration只需一个字符串。请记住,它使用chain of Responsiblity来找到第一个注册的Provider,该Provider知道如何处理url。该应用程序可以与供应商无关,并且可以从属性文件中获取连接URL和驱动程序类名称。这样,如果我更换供应商,则不必重新编译我的代码。但是,如果我对“OracleConnectionWrapper”的引用进行了硬编码,然后更改了供应商,则必须重写部分代码,然后重新编译。关于java - 没有提供者的服务提供者接口(interface),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11376508/
我正在尝试使用ruby和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我
我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..
我好像记得Lua有类似Ruby的method_missing的东西。还是我记错了? 最佳答案 表的metatable的__index和__newindex可以用于与Ruby的method_missing相同的效果。 关于ruby-难道Lua没有和Ruby的method_missing相媲美的东西吗?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/7732154/
我有一个奇怪的问题:我在rvm上安装了rubyonrails。一切正常,我可以创建项目。但是在我输入“railsnew”时重新启动后,我有“程序'rails'当前未安装。”。SystemUbuntu12.04ruby-v"1.9.3p194"gemlistactionmailer(3.2.5)actionpack(3.2.5)activemodel(3.2.5)activerecord(3.2.5)activeresource(3.2.5)activesupport(3.2.5)arel(3.0.2)builder(3.0.0)bundler(1.1.4)coffee-rails(
我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
最近,当我启动我的Rails服务器时,我收到了一长串警告。虽然它不影响我的应用程序,但我想知道如何解决这些警告。我的估计是imagemagick以某种方式被调用了两次?当我在警告前后检查我的git日志时。我想知道如何解决这个问题。-bcrypt-ruby(3.1.2)-better_errors(1.0.1)+bcrypt(3.1.7)+bcrypt-ruby(3.1.5)-bcrypt(>=3.1.3)+better_errors(1.1.0)bcrypt和imagemagick有关系吗?/Users/rbchris/.rbenv/versions/2.0.0-p247/lib/ru
在Rails4.0.2中,我使用s3_direct_upload和aws-sdkgems直接为s3存储桶上传文件。在开发环境中它工作正常,但在生产环境中它会抛出如下错误,ActionView::Template::Error(noimplicitconversionofnilintoString)在View中,create_cv_url,:id=>"s3_uploader",:key=>"cv_uploads/{unique_id}/${filename}",:key_starts_with=>"cv_uploads/",:callback_param=>"cv[direct_uplo
我想在Ruby中创建一个用于开发目的的极其简单的Web服务器(不,不想使用现成的解决方案)。代码如下:#!/usr/bin/rubyrequire'socket'server=TCPServer.new('127.0.0.1',8080)whileconnection=server.acceptheaders=[]length=0whileline=connection.getsheaders想法是从命令行运行这个脚本,提供另一个脚本,它将在其标准输入上获取请求,并在其标准输出上返回完整的响应。到目前为止一切顺利,但事实证明这真的很脆弱,因为它在第二个请求上中断并出现错误:/usr/b
您如何在Rails中的实时服务器上进行有效调试,无论是在测试版/生产服务器上?我试过直接在服务器上修改文件,然后重启应用,但是修改好像没有生效,或者需要很长时间(缓存?)我也试过在本地做“脚本/服务器生产”,但是那很慢另一种选择是编码和部署,但效率很低。有人对他们如何有效地做到这一点有任何见解吗? 最佳答案 我会回答你的问题,即使我不同意这种热修补服务器代码的方式:)首先,你真的确定你已经重启了服务器吗?您可以通过跟踪日志文件来检查它。您更改的代码显示的View可能会被缓存。缓存页面位于tmp/cache文件夹下。您可以尝试手动删除