问题
Question 1 - Basic LUBConstraints
My first try playing around with existing LUBConstraints fails for missing evidence (see code block below). Any hint why? Isn't an empty list a valid list of longs? no element violates the constraint.
import shapeless.ops.coproduct
import shapeless.{::, :+:, Coproduct, HNil, HList}
object testLUBConstraints {
import shapeless.LUBConstraint._
// !!! see comment on question - this satisfies the implicit below!!!
// implicit val hnilLUBForLong = new LUBConstraint[HNil.type, Long] {}
def acceptLong[L <: HList : <<:[Long]#λ](l: L) = true
val validLong = acceptLong(1l :: HNil)
val validEmpty = acceptLong(HNil)
// => WHY??? Error: could not find implicit value for evidence parameter of type shapeless.LUBConstraint[shapeless.HNil.type,Long]
// MY EXPECTATION WAS: 'implicit def hnilLUB[T] = new LUBConstraint[HNil, T] {}' defined within LUBConstraint companion should provide so
// val invalid = acceptLong(1.0d :: HNil) -> fails due to missing evidence (as expected)
}
Any help appreciated.
Question 2 - Own Constraint using Coproduct (split into a seperate question: Shapeless: own HList constraint using Coproduct)
Question 3 - restrict case classes by parameter types (split into a separat question: Shapeless: restricting case class types)
回答1:
The inferred type of the value HNil
will be the singleton type HNil.type
, which is a proper subtype of HNil
. Because type classes like LUBConstraint
are invariant, any available instance for HNil
won't be found if you're asking for an instance for HNil.type
.
There's been some discussion of changing the definition of HNil
so that this would work, but it's not trivial, and it's not clear that all the implications of the change are desirable. In the meantime you can write HNil: HNil
to upcast the value.
来源:https://stackoverflow.com/questions/32778746/shapeless-trying-to-restrict-hlist-elements-by-their-type