我正在开发一个“服务提供商”(与 Google Juice 非常相似)。 它的工作是将任意class_1或接口(interface)映射到可以实例化的*class_2*,并且实现class_1(如果class_1是一个接口(interface))或者是/扩展class_1(如果class_1)不是一个接口(interface))。
所以目前我有方法
public <T> void map(Class<T> key, Class<? extends T> service)
代码的问题是我可以将接口(interface)类型映射到接口(interface)类型,例如
ServiceProvider sp = new ServiceProvider();
sp.map(IParent.class, IChild.class);
这可以完美编译,没有错误(IChild 扩展了 IParent)。 当然,稍后当我想要时:
IParent obj = sp.getService(IParent.class); //look for IParent mapping and instantiate the proper object
我收到 java.lang.InstantiationException。
所以问题是:
如何声明map()方法,以便编译器检查第二个参数是类实现或扩展第一个参数并且不是接口(interface)? (重点是编译时错误。我已经知道如何在运行时检查这一点)
谢谢。 PS:是的——用谷歌搜索了很多,几乎什么也没找到。
更新: 感谢大家抽出时间。 也许再多说几句来说明这一切是怎么回事: 目标是开发机制(ServiceProvider 类),这将帮助我避免应用程序中的紧密耦合和硬依赖。 所以我制定/采用了以下理念: “如果您想交付一个‘单元’(可重用软件的一部分)——声明所有公共(public)接口(interface); 实现是你的私事,我们根本不关心; 如果有人想使用您的设备,他们应该通过以下方式向服务提供商请求: 1. 在启动时,他们使用 sp.map(IYuorInterface.class, TheClassImplementingIt.class); 2. IYuorInterface obj = sp.getService(IYuorInterface.class); 您负责使 TheClassImplementingIt 类“可实例化”(构造函数、安全性、类不是抽象的等);如果没有——获得运行时异常是完全可以的。
有什么好处?
这是非常明显的——在任何给定时间任何其他开发人员都可以重新实现 IYuorInterface 将其映射到 ServiceProvider,然后所有应用程序都将使用它而无需更改一行代码。 (需要时最简单的情况——单元测试)
所以我的观点/问题是:map()方法中的第二个参数必须是代表类的Class类型,而不是接口(interface),并且它应该与第一个参数“分配兼容”。
换句话说,当我执行 map(ISomeIterface.class, Something.class); 时Something 的对象(实例)应该像这样使用:
ISomeIterface obj = sp.getService(ISomeIterface.class);
//or in other words, just for example -- the Something can be used like this:
ISomeIterface obj = new Something();
这就是为什么工厂(正如一些答案所建议的那样)是 Not Acceptable ,ServiceProvider 类已经是一种“工厂”
<小时/>..看起来Java的Class对象既代表类又代表接口(interface),并且在编译时无法区分是真正的类还是接口(interface)..
但无论如何 - 感谢所有人。
最佳答案
您可以使用
找出一个类是否是一个具体的、可实例化的类public boolean isConcrete(Class<?> input)
{
if (input.isInterface())
return false;
if (Modifier.isAbstract(input.getModifiers()))
return false;
return true;
}
关于java - 在java中传递接口(interface)类型(不实现类类型)作为参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16056251/