如何解决指定较弱的 Aux 类型绑定不会导致编译错误
这是一个例子:
sealed trait Type
object Type {
case object Type1 extends Type
case object Type2 extends Type
}
sealed trait TypedTrait {
type Tpe <: Type
}
object TypedTrait {
type Aux[T <: Type] = TypedTrait{ type Tpe = T }
}
sealed trait Test {
//Bounded with TypedTrait.Aux[Type.Type1.type]
type TptTrait <: TypedTrait.Aux[Type.Type1.type]
}
object Test {
//No TypedTrait.Aux[Type.Type1.type] type bound
type Aux[T <: TypedTrait] = Test { type TptTrait = T } //compiles fine,but I expected error
}
该示例针对 Scala 2 和 Scala 3 进行编译,但似乎没有意义。为什么要编译?
解决方法
在编译时,您不能使用错误类型实例化此类型:
class Wololo extends TypedTrait { type Tpe = Type.Type2.type } // bad type
val ax: Test.Aux[Wololo] = new Test { type TptTrait = Wololo }
这失败了
cmd1.sc:24: incompatible type in overriding
type TptTrait <: ammonite.$sess.cmd1.TypedTrait.Aux[ammonite.$sess.cmd1.Type.Type1.type] (defined in trait Test);
found : ammonite.$sess.cmd1.Wololo
required: <: ammonite.$sess.cmd1.TypedTrait.Aux[ammonite.$sess.cmd1.Type.Type1.type]
(which expands to) <: ammonite.$sess.cmd1.TypedTrait{type Tpe = ammonite.$sess.cmd1.Type.Type1.type}
val ax: Test.Aux[Wololo] = new Test { type TptTrait = Wololo }
^
Compilation Failed
同时
class Wololo extends TypedTrait { type Tpe = Type.Type1.type } // good type
val ax: Test.Aux[Wololo] = new Test { type TptTrait = Wololo }
成功
ax: Test.Aux[Wololo] = ammonite.$sess.cmd1$$anon$1@78de58ea
因此我们可以得出结论,这里的边界组合在一起,但在您实际尝试创建一个违反约束的实例之前不会失败。可能这不会触发规范让它立即失败的任何情况(毕竟你可以创建一个有效的实例!),所以检查/证明被推迟到你尝试创建一个实际的实例。
(由于 sealed trait
s,我在 Ammonite 中将其作为一个整体进行了评估)
@ {
sealed trait Type
object Type {
case object Type1 extends Type
case object Type2 extends Type
}
sealed trait TypedTrait {
type Tpe <: Type
}
object TypedTrait {
type Aux[T <: Type] = TypedTrait{ type Tpe = T }
}
sealed trait Test {
//Bounded with TypedTrait.Aux[Type.Type1.type]
type TptTrait <: TypedTrait.Aux[Type.Type1.type]
}
object Test {
type Aux[T <: TypedTrait] = Test { type TptTrait = T }
}
class Wololo extends TypedTrait { type Tpe = Type.Type1.type }
val ax: Test.Aux[Wololo] = new Test { type TptTrait = Wololo }
}
defined trait Type
defined object Type
defined trait TypedTrait
defined object TypedTrait
defined trait Test
defined object Test
defined class Wololo
ax: Test.Aux[Wololo] = ammonite.$sess.cmd1$$anon$1@78de58ea
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。