我正在用Java/Kotlin实现一个学习代理。此代理的部分功能包括搜索大量可能的选项。有多种搜索可能性空间的好方法,我经常改变主意,决定哪一个是最好的。因此,我决定将其作为一个策略模式来实现。
class Agent(val searchStrategy : SearchStrategy){
fun search(input : InputGraph) : Result{
return searchStrategy.search()
}
}
interface SearchStrategy{
fun search(input : InputGraph) : Result
}
class FastSearch : SearchStrategy{
//implementation here
}
class AccurateSearch : SearchStrategy{
// implementation here
}
class ExperimentalSerach : SearchStrategy{
// implentation here
}最近,我决定运行一个大型的实验套件,以测试各种系统参数的有效性。这是通过一个python脚本完成的,该脚本通过使用不同的config.json文件作为参数运行编译后的jar来启动每个实验。类似于:
{
"numSamples" : 5000,
"environmentDifficulty" : 3,
"hazardProbability" : 0.4,
//etc..
}现在,我想让实验者也能够配置代理使用的策略。做这件事最好的方法是什么?我的第一个想法是,我可以只向config.json添加一个额外的字符串字段:
{
"numSamples" : 5000,
"environmentDifficulty" : 3,
"hazardProbability" : 0.4,
"agentStrategy": "FastSearch"
}然后在主系统中使用when或if分支构建它:
val searchStrategy = when(config.agentStrategy){
"FastSearch" -> FastSearch()
"AccurateSearch" -> AccurateSearch()
"ExperimentalSearch" -> ExperimentalSearch()
val agent = agent(searchStrategy)但是,如果我开始添加更多的策略,这似乎会开始变得尴尬/难以维护。有没有更好的方法?
发布于 2016-09-14 18:07:33
解决这个问题的一种方法是使用类名来加载并创建策略的实例,如下所示:
val agentClass = classLoader.loadClass(config.agentStrategy)!!
val agent = agentClass.newInstance() as SearchStrategy另一种方法是注册所有搜索策略,然后使用给定的config进行匹配,如下所示:
class SearchStrategies(val strategies: List<SearchStrategy>){
fun findForConfig(config:Config) = strategies.find { it.javaClass.name.contains(config.agentStrategy) }
}
//somewhere at boot time
val searchStrategies = SearchStrategies(listOf(FastSearch(), AccurateSearch()))
//when needed
val config = ...
val agent = searchStrategies.findForConfig(config)最后,还可以利用SPI来实现更大的可扩展性和标准遵从性,但代价是复杂性。
https://stackoverflow.com/questions/39487535
复制相似问题