4

这是一个 OO 设计 Q。我在我的应用程序中使用类型安全配置。Config 接口非常有用,但是在我的应用程序中有几个字段;配置文件是必需的。我想做的是创建一个 Config 的子接口并添加这两个顶级方法。像这样的东西

trait AppConfig extends Config{
    def param1:String 
    def param2:String 
}

但是,在给定 Config 实例的情况下创建 AppConfig 的真实实例似乎并不可行。(我不想创建包装器对象并复制 Config 接口上的所有方法)。理想情况下,我正在寻找可以实现接近此目标的东西

val conf:Config = //get config object from somewhere
return conf with AppConfig { overrider def param1 = {"blah"} }

我明白最后一行是无效的。但我正在寻找具有等效功能的模式/结构。

4

3 回答 3

2

我们一直在使用Configrity来处理这样的事情。这是一个例子:

我们将用于单元测试/等的编译默认值保留在对象中

object DefaultConfigs {
  val defaultConfig = Configuration(
    "param1" -> "blah"
  )

  val otherConfig = Configuration(
    "param2" -> "blah"
  )

  val appConfig = defaultConfig.include(otherConfig)
}

然后在运行时我们可以包含或不包含它们

val appConfig = Configuration.load(pathToConfig) include DefaultConfigs.appConfig
于 2012-09-06T22:16:51.003 回答
1

您正在寻找的基本上是一些人所说的“动态混合”。scala 不支持开箱即用。有人开发了一个编译器插件来支持这个:http ://www.artima.com/weblogs/viewpost.jsp?thread=275588

但是它有点旧,该项目似乎不再活跃。

一个更好的选择是使用 scala 宏来实现这个特性(需要 scala 2.10,它还没有稳定的版本)。

不过,在您的情况下,所有这些都可能是矫枉过正。在一些经过良好测试的库可用之前,我建议只手动创建代理(尽管看起来很乏味):

trait ConfigProxy extends Config {
  def impl: Config
  // Forward to the inner config
  def root: ConfigObject = impl.root
  def origin: ConfigOrigin = impl.origin
  //... and so on
}

val conf:Config = //get config object from somewhere
val myConf: AppConfig = new AppConfig with ConfigProxy { 
  val impl = conf
  val param1:String = "foo"
  val param2:String = "bar"
}
于 2012-09-07T09:07:21.280 回答
1

如何结合使用动态和反射。动态处理您的便利方法和反射处理配置上的方法。

这是一个刺:

class ConfigDynamic(val config: Config) extends Dynamic {
  def selectDynamic(name: String)= {
    name match {
      case "field1" => 
        config.getString("field1")
      case x @ _ =>
        // overly simplified here
        val meth = configClass.getDeclaredMethod(x)
        meth.invoke(config)
    }
  }
}
于 2012-09-06T23:03:49.483 回答