当前位置: 首页 > 知识库问答 >
问题:

如何解决Scala中冲突成员的继承问题

易品
2023-03-14

我是Scala初学者。我被告知“特征中的一个领域可以是具体的也可以是抽象的”。

trait T2 {
  val f1: String = "T2f1"
}

trait T3 {
  val f1: String = "T3f1"
}

class C2 extends T2 with T3{}

object Test2 extends App {
  val c2 = new C2
  println(c2.f1)
}

当我运行上述代码时,编译器输出了一些错误消息:

“C2类继承了冲突成员:String类型的特征T2中的值f1和String类型的特征T3中的变量f1(注意:这可以通过在C2类中声明重写来解决。)类C2用T3{}扩展T2

那么,如果C2扩展了具有相同字段名的特征,那么应该改变什么呢?谢谢你的帮助。

共有3个答案

龚振濂
2023-03-14

建议的示例是正确的,但尽量避免循环依赖,这也可能导致一些严重的问题。

这里的问题是,你在两个trait中都有相同的字段,编译器无法解析你要使用哪个trait的哪个属性。因此,您可以通过将其显式地提供给编译器来解决问题。

Trait A{
val x:String=“ABC”}

Trait B extends A{
override val x: String=“DEF”}

class AB extends B{
override val f1: String = super[T3].x}
张宣
2023-03-14

您可以手动解决歧义:

  trait T2 {
    val f1: String = "T2f1"
  }

  trait T3 {
    val f1: String = "T3f1"
  }

  class C2 extends T2 with T3 {
    override val f1: String = "T2f1"
  }

  trait T2 {
    val f1: String = "T2f1"
  }

  trait T3 {
    val f1: String = "T3f1"
  }

  class C2 extends T2 with T3 {
    override val f1: String = "T3f1"
  }

如果是,编译器可以通过线性化自动执行此操作

  trait T2 {
    val f1: String = "T2f1"
  }

  trait T3 extends T2 {
    override val f1: String = "T3f1"
  }

  class C2 extends T2 with T3

Scala特征中的冲突字段

农飞星
2023-03-14

公认的答案是正确的,但请记住,所建议的模式很奇怪,在非平凡的情况下可能会导致难以理解的错误。根据我的经验,重写非抽象VAL只会给你带来麻烦。

问题是初始化代码是定义的类/特征的构造函数的一部分。这意味着代码同时初始化了T2。f1和T3。f1将在创建C2的实例时执行:

trait T2 {
    val f1: String = {
        println("T2")
        "T2f1"
    }
}

trait T3 {
    val f1: String = {
        println("T3")
        "T3f1"
    }
}

class C2 extends T2 with T3 {
    override val f1: String = {
        println("C2")
        "T3f1"
    }
}

new C2 // Will print "T2", then "T3", then "C2"

如果初始化代码有任何重要的副作用,这可能会导致难以追踪错误!它还有一个缺点,就是迫使您在C2中重复一些T3的代码。

如果你不是绝对需要T2。f1和T3。f1要成为VAL,最好使用defs来避免抽象的VAL初始化代码:

trait T2 {
    def f1: String = "T2f1"
}

trait T3 {
    def f1: String = "T3f1"
}

class C2 extends T2 with T3 {
    override val f1: String = "C2f1" // You can keep this a def if you like
}

如果您确实需要f1成为val,例如,如果您需要一个稳定的值来在模式匹配语句中使用它,您可以使用以下选项:

trait T2 {
    val f1: String
    protected def computeF1: String = {
        println("T2")
        "T2f1"
    }
}

trait T3 {
    val f1: String
    protected def computeF1: String = {
        println("T3")
        "T3f1"
    }
}

class C2 extends T2 with T3 {
    override val f1: String = computeF1 // You can keep this a def if you like
    override protected def computeF1: String = super[T3].computeF1
}

new C2 // Only prints "T3" once

最后一个解决方案有点冗长,但它通过避免重写非抽象的val而完全绕过了这个问题。

 类似资料:
  • 问题内容: 我需要一个类的双重继承。我尝试了几种语法,但我不了解元类的概念。 问题答案: 您遇到的问题是,您尝试从其继承的类具有不同的元类: 因此,python无法确定哪个应该是新创建的类的元类。在这种情况下,它必须是一个继承自(或对于较旧的PyQt5版本)和的类。 因此,可以通过显式引入诸如metaclass之类的类来解决元类冲突: 如果需要更详细的描述,那么本文是一个好的开始。 但是,我并不真

  • 问题内容: //这是我的代码,我正在代理工作… //以下错误出现在我的控制台上 我正在使用以下罐子 问题答案: 该班已搬迁到另一个包中的HttpClient 4.4。 从4.4开始 将您的httpcore版本升级到至少4.4即可解决该问题。

  • C STLunordered_map如何解决冲突? 看着http://www.cplusplus.com/reference/unordered_map/unordered_map/,它说“唯一的键容器中的两个元素不能有相同的键。” 这意味着容器确实在解决碰撞。然而,那一页并没有告诉我它是如何做到的。我知道一些解决冲突的方法,比如使用链表和/或探测。我想知道的是c STL无序_映射是如何解决它的。

  • Windows 用tutorial进行的操作 若要进行pull操作,请右击tutorial目录,并选择‘拉取’。 用tutorial进行的操作 在以下画面点击‘确定’。 用tutorial进行的操作 我们看到画面上的警告信息表示自动合并失败。请点击‘关闭’以退出窗口。 用tutorial进行的操作 若您确认变更,请点击‘Yes’。 用tutorial进行的操作 TortoiseGit告诉我们:因"

  • 在上一个页面我们提及到,执行合并即可自动合并Git修改的部分。但是,也存在无法自动合并的情况。 如果远程数据库和本地数据库的同一个地方都发生了修改的情况下,因为无法自动判断要选用哪一个修改,所以就会发生冲突。 Git会在发生冲突的地方修改文件的内容,如下图。所以我们需要手动修正冲突。 ==分割线上方是本地数据库的内容, 下方是远程数据库的编辑内容。 如下图所示,修正所有冲突的地方之后,执行提交。

  • 解决冲突 CVS使用内联“冲突标志”来标记冲突,并且在更新时打印C。历史上讲,这导致了许多问题,因为CVS做得还不够。许多用户在它们快速闪过终端时忘记(或没有看到)C,即使出现了冲突标记,他们也经常忘记,然后提交了带有冲突标记的文件。 Subversion通过让冲突更明显来解决这个问题,它记住一个文件是处于冲突状态,在你运行svn resolved之前不会允许你提交修改,详情见“解决冲突(合并别人