如何用圆点形式的元组大小写的窄字符串类型头编写匹配类型模式?

问题描述 投票:0回答:1

我目前正在尝试感受 Scala 3/dotty 中的新功能。所以我正在尝试重做一些我之前尝试过的无形的东西。给定一个狭窄字符串类型的异构列表(在无形中它是

"a" :: "c" :: "f" :: HNil
,据我所知,在点状中,可以利用元组
("a", "c", "f")
),我想根据一些映射来替换类型。例如,考虑以下伪代码:

type MyListOfNames = ("a", "c", "f")
type Mapping = ("a" -> "b", "c" -> "d")
// somehow apply the mapping/replacements as the new type alias `MyListOfRenamedNames`
type MyListOfRenamedNames = ("b", "d", "f")

为此,我想出了以下代码。重新映射单个缩小的字符串类型正在起作用。但我也无法让它与元组一起工作:

object A:
  trait Remapping
  case object ReEmpty extends Remapping
  case class ReCons[N1 <: String, N2 <: String, R <: Remapping](n1: N1, n2: N2, rest: R) extends Remapping

  type Remapped[X <: String, R <: Remapping] <: String = R match
    case ReEmpty.type     => X
    case ReCons[X, n, _]  => n
    case ReCons[_, _, rr] => Remapped[X, rr]

  type AllRemapped[T <: Tuple, R <: Remapping] <: Tuple = T match
    case Unit      => Unit
    case s *: rest => s match
      case String => Remapped[s, R] *: AllRemapped[rest, R]
  //this part doesn't compile, giving following compile error:
  //type s doesn't satisfy upper bound String

  @main def main: Unit =
    type RemapAtoBAdCtoD = ReCons["a", "b", ReCons["c", "d", ReEmpty.type]]
    val expectedToCompile1: Remapped["a", RemapAtoBAdCtoD] = "b"
    val expectedToCompile2: Remapped["c", RemapAtoBAdCtoD] = "d"
    val expectedToCompile3: Remapped["f", RemapAtoBAdCtoD] = "f"
    val expectedToCompile4: Remapped["a", ReEmpty.type] = "a"
    //above examples compile as expected

    // val expectedNotToCompile: Remapped["a", RemapAtoBAdCtoD] = "a"
    //above example doesn't compile as expected

    //I am trying to get following:
    type MyList = ("a", "c", "f")
    val remapped: AllRemapped[MyList, RemapAtoBAdCtoD] = ("b", "d", "f")
  end main

end A

我得到的编译错误是以下行中的

Type argument s does not conform to upper bound String

      s match
        case String => Remapped[s, R] *: AllRemapped[rest, R]

我使用了 dotty 版本

0.18.1-RC1
,因为它是 Scastie 上最新的可用版本。您可以尝试以下链接:https://scastie.scala-lang.org/BKzhEV7PRiKyfQ3CE2vjww

这是否不受支持,有没有办法实现这一点,即如何进一步限制匹配类型内的类型模式中的类型(我尝试过

case (s <: String) *: rest =>
,但编译器失败并出现错误:
scala.MatchError: Parens(Ident(s)) (of class dotty.tools.dotc.ast.untpd$Parens)
)?还有更好的方法来实现我总体上尝试做的事情(在 dotty 当前的能力范围内,例如
erased
inline
)?

scala pattern-matching scala-3 dotty match-types
1个回答
4
投票

尝试引入辅助类型并将其用作类型模式

type Hlp[X <: String, Rest <: Tuple] = X *: Rest

type AllRemapped[T <: Tuple, R <: Remapping] <: Tuple = T match {
  case Unit         => Unit
  case Hlp[s, rest] => Remapped[s, R] *: AllRemapped[rest, R]
}

inline
erased
不适用于
type

实际上,对于元组的映射,有标准类型

Tuple.Map
,尽管目前在0.18.1-RC1中我无法使其工作

type AllRemapped[T <: Tuple, R <: Remapping] = Tuple.Map[T, [X <: String] =>> Remapped[X, R]]

//Type argument [X <: String] => A.Remapped[X, R] does not conform to upper bound [_$22] => Any 

有了

inline
你就可以做到

inline def g(x: "a" | "c" | "f") <: String = inline x match {
  case "a" => "b"
  case "c" => "d"
  case "f" => "f"
}

g("a"): "b"
g("c"): "d"
g("f"): "f"
// g("x") // doesn't compile

尝试

sealed trait Remapping
case object ReEmpty extends Remapping
case class ReCons[N1 <: String, N2 <: String, R <: Remapping](n1: N1, n2: N2, rest: R) extends Remapping

type Remapped[X <: String, R <: Remapping] <: String = R match {
  case ReEmpty.type     => X
  case ReCons[X, n, _]  => n
  case ReCons[_, _, rr] => Remapped[X, rr]
}

inline def getRemapped[X <: String & Singleton, R <: Remapping] erased (x: X, r: R) <: String = inline r match {
  case ReEmpty             => x
  case rc: ReCons[X, _, _] => rc.n2
  case rc: ReCons[_, _, _] => getRemapped(x, rc.rest).asInstanceOf[Remapped[X, R]]
}

type RemapAtoBAndCtoD = ReCons["a", "b", ReCons["c", "d", ReEmpty.type]]
val remapping: RemapAtoBAndCtoD = ReCons("a", "b", ReCons("c", "d", ReEmpty))
val remapped2: ("b", "d", "f") = (
  getRemapped("a", remapping),
  getRemapped("c", remapping),
  getRemapped("f", remapping)
)  // (b,d,f)

//myList.map[[X <: String] =>> Remapped[X, RemapAtoBAndCtoD]]([X <: String] => (x: X) => getRemapped(x, remapping))
//[error]    |Found:    Object with PolyFunction {...}
//[error]    |Required: PolyFunction{apply: [t](x$1: t): A.Remapped[t, A.RemapAtoBAndCtoD]}
//https://github.com/milessabin/shapeless/pull/901
© www.soinside.com 2019 - 2024. All rights reserved.