7

我正在尝试使用 shapeless 以类型安全的方式轻松地积累对象。

问题是当我想连接(:::)两个时HList。我遇到了一个新手(至少看起来)问题。它错过了Prepend上下文中的隐式实例。

但是,查看hlist.scala,我可以看到泛型implicit def在对象PrependPrependAux.

添加import Prependimport PrependAux手动并没有改变任何东西(显然......)。

所以这里的代码减少到最低限度:

enter code here

import shapeless._
import HList._
import Prepend._
import PrependAux._

object test {

  val a:HList = 1 :: 4 :: "A" :: HNil
  val b:HList = "R" :: false :: HNil

  val c:HList = a ::: b   // <<<<<<<<<<< NEEDS A Prepend in the context 

}

现在在控制台中:

[error]     test.scala:10: could not find implicit value for parameter prepend: shapeless.Prepend[shapeless.HList,shapeless.HList]
[error]     val c:HList = a ::: b   // this needs an implicit Prepend in the current context

什么应该灼伤我的眼睛?

谢谢

编辑

稍微更新一下,把真正的问题重新复杂化了一点,因为之前要粗俗化是要强烈的。

这是我能做的事情:

case class A[L<:HList](a:L) { 
  def doSmth[C <:HList](c:C) = a ::: c 
}

所以我无法访问真正的类型,只有我知道它们是HList

4

1 回答 1

11

向上转换HList是这里的问题。几乎没有什么可以用普通的旧的HList(除了添加新元素)。

您可以提供更多信息的类型注释:

val a: Int :: Int :: String :: HNil = 1 :: 4 :: "A" :: HNil
val b: String :: Boolean :: HNil = "R" :: false :: HNil
val c: Int :: Int :: String :: String :: Boolean :: HNil = a ::: b

或者只是让类型被推断出来,这通常更方便:

val a = 1 :: 4 :: "A" :: HNil
val b = "R" :: false :: HNil
val c = a ::: b

作为对您的评论的回应:如果您确保获得所需的证据,您可以做您想做的事(请注意,我假设这a: A是一个错字a: L,并且您需要这样做-Ydependent-method-types):

case class A[L <: HList](a: L) {
  def doSmth[C <: HList](c: C)(implicit p: Prepend[L, C]) = a ::: c
}

一般来说,您可以只查看您正在使用的操作所必需的隐式,然后将它们包含在您的方法中。

于 2012-06-14T17:00:08.847 回答