首页 文章

使用mixins扩展的Scala不可变容器类

提问于
浏览
0

我想要一个容器类,我可以使用一些特性来扩展,以包含一组默认值,以后可以以不可变的方式进行更改 . 这些特征将保存一些简单的数据,因此创建具有几个特征的类将创建一个具有多个默认值集合的对象 .

然后,我希望能够通过在一次更改一个新值时复制对象来不可修改地修改任何val .

该类可能具有以下内容:

class Defaults(val string: String = "string", val int: Int = "int")

然后像这样的其他特征

trait MoreDefaults{
  val long: Long = 1l
}

然后我想在实例化时混合它们以构建我需要的特定默认值集

var d = new Defaults with MoreDefaults

后来就像:

if (someFlag) d = d.copy( long = 1412341234l )

你可以用一个案例类来做这样的事情,但我在22处没有参数 . 但是我会根据需要使用一堆默认值我想混合,然后允许更改任何一个(以不可变的方式定义类或特征 .

我可以在 Defaults 类中坚持 copy 方法,如下所示:

def copy(
    string: String = string,
    int: Int = int): Defaults = {
  new Defaults(string, int)
}

然后做点什么

var d = new Defaults
if (someFlag) d = d.copy(int = 234234)

Question ====> 这适用于基类中的值,但我无法想象如何将其扩展到mixin特征 . 理想情况下, d.copy 将适用于所有类特征定义的所有val . 重载也很麻烦,因为val主要是字符串,但是所有的val名称在类和特征的任何组合中都是唯一的,或者是一个错误 .

只使用类我可以通过一个基本的Defaults类来获得一些这样的功能,然后用另一个具有它自己的非重载 copyMoreDefault 函数的类来扩展它 . 这真的很丑陋,我希望Scala专家在看到我之前会看到它并且笑得很开心 - 它确实有用 .

class Defaults(
    val string: String = "one",
    val boolean: Boolean = true,
    val int: Int = 1,
    val double: Double = 1.0d,
    val long: Long = 1l) {

  def copy(
      string: String = string,
      boolean: Boolean = boolean,
      int: Int = int,
      double: Double = double,
      long: Long = long): Defaults = {
    new Defaults(string, boolean, int, double, long)
  }   
}

class MoreDefaults(
    string: String = "one",
    boolean: Boolean = true,
    int: Int = 1,
    double: Double = 1.0d,
    long: Long = 1l,
    val string2: String = "string2") extends Defaults (
        string,
        boolean,
        int,
        double,
        long) {

  def copyMoreDefaults(
      string: String = string,
      boolean: Boolean = boolean,
      int: Int = int,
      double: Double = double,
      long: Long = long,
      string2: String = string2): MoreDefaults = {

    new MoreDefaults(string, boolean, int, double, long, string2)
  }

}

然后以下工作:

var d = new MoreDefualts
if (someFlag) d = d.copyMoreDefaults(string2 = "new string2")

如果Defaults得到改变的参数,这个方法将是一团糟!所有派生类都必须更新 - 呃 . 肯定有更好的办法 .

1 回答

  • 1

    我认为我并没有严格地回答你的问题,而是建议另一种解决方案 . 因此,您遇到大案例类的问题,例如

    case class Fred(a: Int = 1, b: Int = 2, ... too many params ... )
    

    我要做的是将params组织成更多案例类:

    case class Bar(a: Int = 1, b: Int = 2)
    case class Foo(c: Int = 99, d: Int = 200)
    // etc
    case class Fred(bar: Bar = Bar(), foo: Foo = Foo(), ... etc)
    

    然后,当你想要复制和更改时,请说出 Foo 的一个值:

    val myFred: Fred = Fred()
    val fredCopy: Fred = myFred.copy(foo = myFred.foo.copy(d = 300))
    

    你甚至不需要定义复制功能,你可以免费获得它们 .

相关问题