从 Shapeless HList 到 case class 的转换中缺少隐式 Generic.Aux
Implicit Generic.Aux missing on conversion from Shapeless HList to case class
我最近才开始学习 scala,今天我决定要编写一个 CSV 解析器,它可以很好地加载到 case classes 中,但将数据存储在 Shapeless 的 HList 对象的行(列表)中,这样我可以接触到类型级编程。
这是我目前的情况:
// LoadsCsv.scala
import shapeless._
import scala.collection.mutable
trait LoadsCsv[A, T <: HList] {
val rows: mutable.MutableList[T] = new mutable.MutableList[T]
def convert(t: T)(implicit gen: Generic.Aux[A, T]): A = gen.from(t)
def get(index: Int): A = {
convert(rows(index))
}
def load(file: String): Unit = {
val lines = io.Source.fromFile(file).getLines()
lines.foreach(line => rows += parse(line.split(",")))
}
def parse(line: Array[String]): T
}
加载数据集的对象:
// TennisData.scala
import shapeless._
case class TennisData(weather:String, low:Int, high:Int, windy:Boolean, play:Boolean)
object TennisData extends LoadsCsv[TennisData, String :: Int :: Int :: Boolean :: Boolean :: HNil] {
load("tennis.csv")
override def parse(line: Array[String]) = {
line(0) :: line(1).toInt :: line(2).toInt :: line(3).toBoolean :: line(4).toBoolean :: HNil
}
}
在我添加 get() 以及从 HList 到案例 class 的转换之前,一切似乎都正常,但我现在遇到编译错误。 为什么不加载隐式,我可以做些什么来修复它或以其他方式从 HList 转换为大小写 class?
Error:(14, 17) could not find implicit value for parameter gen: shapeless.Generic.Aux[A,T]
return convert(rows(index))
^
我一直在阅读 shapeless 文档,它提到这个领域在版本 1 和 2 之间一直在变化,但我相信事情 应该 在我的版本上工作shapeless 和 scala 所以我怀疑我做错了什么。
作为参考,我是 运行 scala 2.11.6 和 shapeless 2.2.2
你非常接近。问题是 Scala 不会自动为您在调用链中传播隐式需求。如果您需要一个 Generic[A, T]
实例来调用 convert
,那么您必须确保每次调用 convert convert
时都在范围内。如果 A
和 T
是固定的(并且实际上是 class-HList
对),Shapeless 将为您生成一个。但是,在您的 get
方法中,编译器仍然对 A
和 T
一无所知,除了 T
是一个 HList
,因此您需要 require 实例再次调用 convert
:
def get(index: Int)(implicit gen: Generic.Aux[A, T]): A = convert(rows(index))
此更改后一切都应该正常工作。
请注意,您还可以通过添加如下(抽象)方法在特征级别要求实例:
implicit def genA: Generic.Aux[A, T]
然后任何 class 实现 LoadsCsv
可以有一个隐含的 val genA
参数(或者可以以其他方式提供实例)。
我最近才开始学习 scala,今天我决定要编写一个 CSV 解析器,它可以很好地加载到 case classes 中,但将数据存储在 Shapeless 的 HList 对象的行(列表)中,这样我可以接触到类型级编程。
这是我目前的情况:
// LoadsCsv.scala
import shapeless._
import scala.collection.mutable
trait LoadsCsv[A, T <: HList] {
val rows: mutable.MutableList[T] = new mutable.MutableList[T]
def convert(t: T)(implicit gen: Generic.Aux[A, T]): A = gen.from(t)
def get(index: Int): A = {
convert(rows(index))
}
def load(file: String): Unit = {
val lines = io.Source.fromFile(file).getLines()
lines.foreach(line => rows += parse(line.split(",")))
}
def parse(line: Array[String]): T
}
加载数据集的对象:
// TennisData.scala
import shapeless._
case class TennisData(weather:String, low:Int, high:Int, windy:Boolean, play:Boolean)
object TennisData extends LoadsCsv[TennisData, String :: Int :: Int :: Boolean :: Boolean :: HNil] {
load("tennis.csv")
override def parse(line: Array[String]) = {
line(0) :: line(1).toInt :: line(2).toInt :: line(3).toBoolean :: line(4).toBoolean :: HNil
}
}
在我添加 get() 以及从 HList 到案例 class 的转换之前,一切似乎都正常,但我现在遇到编译错误。 为什么不加载隐式,我可以做些什么来修复它或以其他方式从 HList 转换为大小写 class?
Error:(14, 17) could not find implicit value for parameter gen: shapeless.Generic.Aux[A,T]
return convert(rows(index))
^
我一直在阅读 shapeless 文档,它提到这个领域在版本 1 和 2 之间一直在变化,但我相信事情 应该 在我的版本上工作shapeless 和 scala 所以我怀疑我做错了什么。
作为参考,我是 运行 scala 2.11.6 和 shapeless 2.2.2
你非常接近。问题是 Scala 不会自动为您在调用链中传播隐式需求。如果您需要一个 Generic[A, T]
实例来调用 convert
,那么您必须确保每次调用 convert convert
时都在范围内。如果 A
和 T
是固定的(并且实际上是 class-HList
对),Shapeless 将为您生成一个。但是,在您的 get
方法中,编译器仍然对 A
和 T
一无所知,除了 T
是一个 HList
,因此您需要 require 实例再次调用 convert
:
def get(index: Int)(implicit gen: Generic.Aux[A, T]): A = convert(rows(index))
此更改后一切都应该正常工作。
请注意,您还可以通过添加如下(抽象)方法在特征级别要求实例:
implicit def genA: Generic.Aux[A, T]
然后任何 class 实现 LoadsCsv
可以有一个隐含的 val genA
参数(或者可以以其他方式提供实例)。