java - 当一个类具有许多依赖项但需要根据某些条件仅使用其中一些依赖项时的设计方法
问题描述
我有多个实现接口并返回对象的类。
public interface DataFetcher {
Data getData(Info info);
}
public class Data {
private String name;
private String value;
}
@Component
public class DataPointA implements DataFetcher {
@Override
public Data getData(Info info) {
//..Do some processing
return new Data("SomeName", valueComputed);
}
}
现在我有大约 20 个数据点,它们实现了 DataFetcher 类并返回数据对象。
我将所有数据点自动连接到一个类,并根据某些条件使用某些数据点。
@Component
public class DataComputer {
@Autowired
private DataPointA dataPointA;
@Autowired
private DataPointB dataPointB;
.
.
.
public void computeData(String inputType, Info info) {
List<DataFetcher> dataFecthers;
switch(inputType) {
case "typeA" : dataFecthers = ImmutableList.of(dataPointA, dataPointB);
break;
.
.
.
case "typeD" : dataFecthers = ImmutableList.of(dataPointE, dataPointF, dataPointG);
break;
}
dataFetcher.forEach(dataPoint -> {
//Do some processing with dataPoint.getData(info)
})
}
}
可以看出,DataComputer 类将有一个完整的依赖项列表,这些依赖项可能变得无法管理。此外,基于 inputType 使用的数据点也是事先知道的,因此可以将其提取出来。这是我的尝试:
@Component
public class DataComputationPointDecider {
@Autowired
private DataPointA dataPointA;
@Autowired
private DataPointB dataPointB;
.
.
.
@Bean
public Map<String, List<DataFetcher>> getDataComputationPoints() {
return new ImmutableMap.Builder<String, List<DataFetcher>>()
.put("typeA", ImmutableList.of(dataPointA, dataPointB))
.put("typeD", ImmutableList.of(dataPointE, dataPointF, dataPointG))
.build();
}
}
然后我的 DataComputer 依赖项减少了:
@Component
public class DataComputer {
@Autowired
private Map<String, List<DataFetcher>> dataComputationPoints;
public void computeData(String inputType, Info info) {
List<DataFetcher> dataFecthers = dataComputationPoints.get(inputType);
dataFetcher.forEach(dataPoint -> {
//Do some processing with dataPoint.getData(info)
})
}
}
有没有更好的方法来设计这个?
解决方案
我认为您的方法没有任何重大错误。但我建议另一种选择。
您可以决定或说出它可以处理哪些输入类型,而不是维护一个映射一个inputType
与一个列表的映射。DataFetcher
DataFetcher
但这需要改变DataFetcher
as的接口
public interface DataFetcher {
boolean canHandle(String inputType);
Data getData(Info info);
}
实现看起来像
@Component
public class DataPointA implements DataFetcher {
@Override
boolean canHandle(String inputType) {
return "typeA".equals(inputType);
}
@Override
public Data getData(Info info) {
//..Do some processing
return new Data("SomeName", valueComputed);
}
}
然后您可以将所有内容DataFetcher
作为一个列表注入(并且不需要@Autowired
为每个列表添加一个字段)并将其处理为
@Autowired
List<DataFetcher> dataFetchers;
...
dataFetchers.stream()
.filter(dataFetcher -> dataFetcher.canHandle(inputType))
.forEach(dataFetcher.getData(info));
优点:
在您当前的方法中,如果您添加一个新的DataFetcher
实现,您需要添加一个@AutoWired
字段/成员并修改 (getDataComputationPoints) 映射。但是,有了这个,DataFetcher
可以处理的 inputTypes 是用它本身指定的,因此您只需要为新的输入类型添加新的类。
参考
更新:
缺点
输入类型是在类中指定的,这意味着您无法轻松找到
DataFetchers
给定输入类型的(数据点)列表。如果您需要删除对 inputType 的支持,那么您需要再次访问每个实现(从 中删除该 inputType
canHandle
)。在您的方法中,它只是简单地删除一个地图条目。
推荐阅读
- jinja2 - 如何动态显示 PIL 图像
- azure - 无法将 esp8266 设备连接到 azure IoT hub
- java - 有什么方法可以在 Java 哈希图中通过键获取特定条目(已插入多个值)?
- java - 时事通讯如何
- ios - 在 viewDidLoad() 中获取视图的渲染边界
- sql - SQL join 大于/小于 then
- docker - Dockerfile FROM --platform 选项
- javascript - 使用 React Context 时如何避免意外渲染?
- unity3d - 从相机到播放器的 RayCastAll 不工作
- java - 如何使用 -add-modules 添加多个模块