加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 综合聚焦 > 服务器 > 安全 > 正文

接受任何在traala中扩展trait作为参数的case类

发布时间:2020-12-16 09:55:34 所属栏目:安全 来源:网络整理
导读:我需要一个函数来接受任何扩展特征的case类.举个例子 假设有一个特征 trait Action { def execute(param: Parameters)} 并且可以传递以下参数,具体取决于它的操作类型 trait Parameterscase class A(x: String,y: Double) extends Parameterscase class B(x:
我需要一个函数来接受任何扩展特征的case类.举个例子

假设有一个特征

trait Action {
    def execute(param: Parameters)
}

并且可以传递以下参数,具体取决于它的操作类型

trait Parameters
case class A(x: String,y: Double) extends Parameters
case class B(x: Int,y:String,z: Double) extends Parameters

用户应该能够指定它想要的参数

class Action1 extends Action {
    override def execute(param: A)
}

class Action2 extends Action {
    override def execute(param: B)
}

在打电话的时候

def run(t: Action) {
    t.execute(new A("a",1.0))
}

怎么做到这一点?当我实现Action1时,它表示我将得到我们无法覆盖的错误,因为trait说execute方法接受Parameters,但是在重写时,我们需要一个扩展Parameters的类.

解决方法

你需要考虑的是为什么你需要你的Action来开始执行方法.可能你的代码中有其他类似的东西:

def executeAll(actions: Seq[Action],params: Parameters) = 
    actions.foreach { _.execute(params) }

请注意,此处您的操作列表可以包含不同类型的操作.如果其中一些除了类型A的参数,并且有些需要它是B类,它将无法工作,因为你的params参数不能同时存在.

出于这个原因,你不能覆盖一个方法来在子类中采用不同的参数类型(实际上,你可以,但是类型必须是原始类型的超类,而不是子类.他们说函数是逆变的这个reas的参数类型,但我离题了.

如果Action1和Action2实际上可以实际处理它们的参数,通过trait接口而不要求它们是特定类型,那么解决方案很简单:只需将覆盖方法的声明更改为覆盖def execute(参数:参数) ).

如果不是这种情况,并且Action1只能处理类型A的参数,那么(1)可能(虽然,不一定)你的设计有问题,(2)你必须使用参数化解决方案,正如另一个答案所示.但是你必须改变所有处理Action的地方:

def executeAll[T <: Parameters](actions: Seq[Action[T]],params: T) = ...

此声明约束params参数的类型以匹配列表中的操作所期望的内容,因此不会错误地使用B调用Action1.execute.

现在,正如你在其他答案的评论中提到的那样,你不能混合不同类型的动作,因为它没有意义:它们本质上是不同的类.如果您不打算使用execute方法,并且只想传递一些代码周围的不同操作列表,则可以使用通配符语法:

val actions: List[Action[_] = List(Action1(...),Action2(...))

同样,此列表可以作为任何类型的操作的通用容器,但是您不能使用它来一般地应用execute方法,因为这将要求参数同时具有不同类型.

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读