0

首先我不知道如何称呼我的问题,所以任何关于更好标题的建议请告诉我。

我想创建一个接口,以便其他人可以添加他们的实现,然后在运行时决定使用哪个实现。

即我创建了一个接口“Food”,并且存在 2 个实现“Breakfast”和“Lunch”。在实现中,两者都有相同的方法名称和相同的参数,我想让用户在运行时调用它们中的任何一个。“makeFood() 使用早餐”或“makeFood() 使用午餐”。

一种可能性是只编译一个实现,但如果将来必须使用另一个实现,这将需要删除 .class 文件。

或者我应该使用某种类型的继承?

另外(不是优先级):我不知道实现类名称,所以有人可以创建“晚餐”或“secondBreakfast”。这会让我陷入另一个困境,如果用户请求一个尚未实现的“类”该怎么办:SI 总是可以假设用户会知道哪些实现是可用的。

4

3 回答 3

5

在我看来,这听起来像是策略模式将满足您的需求(GOF - http://en.wikipedia.org/wiki/Strategy_pattern)。

我不清楚你希望它有多动态。拥有一个接口的两个实现和一个选择器方法是最简单的。但是您也可以动态地动态生成接口的实现,这可能对您来说太过分了。无论哪种方式,策略模式都可以抽象出这种复杂性,并使您能够根据所需的任何运行时标准选择不同的行为。

这是一个动态加载类的示例,假设您已经知道完全限定的类名并且该对象具有无参数构造函数:

Class c = Class.forName("java.lang.Object");
Object o = c.newInstance();

System.out.println( "o = " + o );

对于这种情况,您需要捕获的错误是:InterruptedException、ClassNotFoundException、IllegalAccessException、InstantiationException;很多,但只是以相同的方式处理它们并拒绝用户的选择。

如果你需要一个带参数的构造函数,那么:

Class c = Class.forName("java.lang.String");
Constructor cons = c.getConstructor( String.class ); // the args here are the expected types for the constructor that you require on the class
String s = (String) cons.newInstance( "hello" );

这将添加更多必须捕获的异常:InterruptedException、ClassNotFoundException、IllegalAccessException、InstantiationException、NoSuchMethodException、InvocationTargetException。但是再次以与以前相同的方式拒绝用户选择。

于 2013-04-12T07:59:07.693 回答
1

如果您不知道类名并想知道所有可用的类,则需要在运行时获取接口的所有实现。这可以通过反射来实现:http ://code.google.com/p/reflections/

看看方法getSubTypesOf

如果您希望用户提供自己的实现并使用这个,您有不同的选择:

1) 在启动时阅读提供的 Jar 并使用带有预配置设置的依赖注入(例如使用 Spring)。用户可以在应用程序启动之前将完全合格的类名写入设置。

2)使用插件机制,如 ServiceLoader:http ://docs.oracle.com/javase/6/docs/api/java/util/ServiceLoader.html 或 JSPF:http ://code.google.com/p /jspf/

或者,如果您想在运行时阅读新的实现,您可以使用 OSGi:http ://en.wikipedia.org/wiki/OSGi (看看 Apache Karaf、Eclipse Equinox 和 Apache Felix)。

于 2013-04-12T08:37:39.150 回答
0

您似乎在准确描述 Java 的interface. 例如:

食物.java:

interface Food {
    void makeFood();
}

早餐.java:

class Breakfast implements Food {
    void makeFood() { System.out.println("Breakfast."); }
}

午餐.java:

class Lunch implements Food {
    void makeFood() { System.out.println("Lunch."); }
}

Food正如您所说,在这里我们有两种不同的实现。现在,我们可以Food通过构造任何一个实现类来实例化一个对象:

Food myFood = new Breakfast();

或者

Food myFood = new Lunch();

没有删除文件或类似的东西。希望这能让事情更清楚。

于 2013-04-12T07:59:59.640 回答