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

scala – 模式匹配依赖类型 – 如何避免asInstanceOf?

发布时间:2020-12-16 18:31:18 所属栏目:安全 来源:网络整理
导读:我在如何完成以下操作时没有作弊和使用asInstanceOf. 假设我有一些任意密封类型的对象,每个对象都有自己的类型成员. sealed trait Part { type A } case object P1 extends Part { override type A = String } case object P2 extends Part { override type
我在如何完成以下操作时没有作弊和使用asInstanceOf.

假设我有一些任意密封类型的对象,每个对象都有自己的类型成员.

sealed trait Part { type A }
  case object P1 extends Part { override type A = String }
  case object P2 extends Part { override type A = Int }

现在说我将P和P.A值捆绑在一起……

trait PartAndA {
    val p: Part
    val a: p.A
  }

  object PartAndA {
    type Aux[P <: Part] = PartAndA {val p: P}

    def apply(_p: Part)(_a: _p.A): Aux[_p.type] =
      new PartAndA {
        override val p: _p.type = _p
        override val a          = _a
      }
  }

如何通过耗尽检查和没有手动演员安全地完成以下操作?

def fold[A](pa: PartAndA)(p1: PartAndA.Aux[P1.type] => A,p2: PartAndA.Aux[P2.type] => A): A =
    pa.p match {
      case P1 => p1(pa.asInstanceOf[PartAndA.Aux[P1.type]])
      case P2 => p2(pa.asInstanceOf[PartAndA.Aux[P2.type]])
    }

解决方法

我认为你的问题与 jvm type erasure有关.没有它你的问题可以简化为:

sealed trait Part { type A }
case class P1() extends Part { override type A = String }
case class P2() extends Part { override type A = Int }

trait PartAndA[P <: Part] {
  val p: P
  val a: p.A
}

object PartAndA {
  type Aux[P <: Part] = PartAndA[P]

  def apply(_p: Part)(_a: _p.A): PartAndA[_p.type] =
    new PartAndA[_p.type] {
      override val p: _p.type = _p
      override val a          = _a
    }
}

def fold[A,T: ClassTag](pa: PartAndA[T])(p1: PartAndA[P1] => A,p2: PartAndA[P2] => A): A =
  pa match {
    case s: PartAndA[P1]  => p1(pa) // here P1 is lost,err 
    case i: PartAndA[P2] => p2(pa) // here P2 is lost,err
  }

根据我的知识,jvm类型擦除没有更短(比你的或typeTags/classTags)的解决方法.

(编辑:李大同)

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

    推荐文章
      热点阅读