我目前正在尝试了解 Scala 3/dotty 中的新功能。所以我想重做一些我以前用 shapeless 尝试过的东西。"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")
为此,我想出了以下代码。重新映射单个缩小的 String 类型正在工作。但我也无法让它与元组一起使用:
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
and inline
)?