我有一个Data
类和几个子类,如JSONData
、XMLData
、IntegerData
。任务是处理不同类型的传入数据。因此,基于程序到接口,而不是实现,我创建了以下接口和泛型类型用于编译时类型检查:
interface DataProcessor<T extends Data> {
void process(T data);
}
有几种基于此接口的实现:
* `class JSONDataProcessor implements DataProcessor<JSONData>`
* `class XMLDataProcessor implements DataProcessor<XMLData>`
* `class IntegerDataProcessor implements DataProcessor<IntegerData>`
剩下的工作是创建一个简单工厂,用于创建相应的DataProcessor
实例。因此,我创建了以下简单的工厂,或者说它实际上只是一个处理器映射器,因为具体的处理器可以在ProcessorFactory中缓存为静态变量:
public class ProcessorFactory {
public static DataProcessor<?> create() {
//logic of return an instance
}
}
上面的设计有一个问题-不能直接调用返回的DataProcessor
实例上的process
方法:
Data data = jsonData;
ProcessorFactory.create().process(data);
问题:由于数据必须是Data
的具体子类,因此编译时键入检查导致上面的代码出现编译错误,如何解决此问题?还是设计本身不好?如果是这样的话,还有什么更好的设计呢?
虽然设计模式很酷,但您在问题中报告的编译错误并不是由于缺乏双重调度造成的。
您会得到编译错误,因为例如,通过声明this:JSONDataProcessor implements DataProcessor<JSONData>{...}
,您已经声明了this:void process(JSONData data)
。
您可能假设<T extends Data>
意味着您可以将静态类型为Data
的对象实例传递到void process(JSONData data)
中,因为毕竟是Data extends Data
。但Java不是这样工作的。
查看编译错误原因的一种方法是考虑一个方法声明,如:public static void main(String arg){...}
。即使String extends Object
,将静态类型为Object
的引用传递到声明为main(String)
方法中也是非法的。
如果您尝试将该方法调用为main(new Object())
,则会出现与DataProcessor
相同的编译错误。通过引入不必要的设计模式来纠正您犯下的错误是过分的。在您的情况和main(String)
的情况下,最简单的校正是传入方法声明要采用的类型。
">…如何解决此问题">
在我看来,最简单的解决方案是按照最初声明方法的方式使用它们。如果你的实现方式与我的类似,那么我已经确认这是有效的…
...
JSONData data = new JSONData( ... );
ProcessorFactory.create().process(data);
...
这也适用于我的演示(不需要设计模式(…
DataProcessor< Data< ? > > dProc = DataProcessor.Factory.create( );
Data<String> data = new JSONData( ... );
dProc.process( data );
">…设计本身是否糟糕">
将设计称为">好"或">坏是主观的。更客观的做法是问自己:设计正确吗?它正确地做了你打算做的事情吗?如果它做了你打算做的事,那么它是正确的。如果没有,那么回到白色绘图板。
另一个设计选项是决定根本不使用泛型--,也不使用设计模式。你可能只需要简单一点的东西。
你提到:">编程到接口"。也许你所有的设计需求都是老式接口形式的老式子类型多态性。泛型可能不是你想要做的最好的设计选择。
这是Java的经典问题,因为它不支持双重调度。人们使用访问者模式规避了这个问题。在您的情况下,您可以在接受DataProcessor
的Data
类中公开一个访问函数,并运行它的process
方法。
像这样的
interface Data {
....
void visit(DataProcessor processor);
}
Data d = JsonData;
d.visit(jsonDataProcessor processor);
JsonData的visit
函数看起来像
void visit(JsonDataProcessor processor) {
processor.process(this);
}