我刚刚开始学习scala,今天我决定编写一个CSV解析器,它可以很好地加载到case类中,但是将数据存储在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
}
}
事情似乎工作正常,直到我添加了从HList到case类的get(),我现在得到了一个编译错误。 为什么不隐式加载,我该怎么做才能修复它或以其他方式从HList转换为案例类?
Error:(14, 17) could not find implicit value for parameter gen: shapeless.Generic.Aux[A,T]
return convert(rows(index))
^
我一直在阅读没有形状的文档,它提到这个区域在版本1和版本2之间不断变化,但我相信应该正在处理我的无形和scala版本所以我怀疑我做错了什么。
作为参考,我正在运行scala 2.11.6和无形2.2.2
答案 0 :(得分:5)
你非常接近。问题是Scala不会自动为调用链传播隐式需求。如果您需要Generic[A, T]
个实例来呼叫convert
,那么每次调用转化convert
时,您都必须确保其范围。如果A
和T
已修复(并且实际上是一个案例类 - HList
对),Shapeless将为您生成一个。但是,在get
方法中,除了A
是T
之外,编译器仍然对T
和HList
一无所知,因此您需要要求实例再次拨打convert
:
def get(index: Int)(implicit gen: Generic.Aux[A, T]): A = convert(rows(index))
在这次改变之后,一切都应该正常工作。
请注意,您还可以通过添加如下所示的(抽象)方法来要求特征级别的实例:
implicit def genA: Generic.Aux[A, T]
然后任何实现LoadsCsv
的类都可以有一个隐式val genA
参数(或者可以用其他方式提供实例)。