从Shapeless HList到case类的转换时缺少Implicit Generic.Aux

Rut*_*ulV 3 scala type-level-computation shapeless

我刚刚开始学习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

}
Run Code Online (Sandbox Code Playgroud)

并且正在加载数据集的对象:

// 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
    }

}
Run Code Online (Sandbox Code Playgroud)

事情似乎工作正常,直到我添加了从HList转换到案例类的get(),现在我得到了编译错误.为什么不隐式获取加载,我该怎么做才能修复它或以其他方式从HList转换为case类?

Error:(14, 17) could not find implicit value for parameter gen: shapeless.Generic.Aux[A,T]
        return convert(rows(index))
                      ^
Run Code Online (Sandbox Code Playgroud)

我一直在阅读无形文档,它提到这个区域在版本1和版本2之间不断变化,但我相信事情应该在我的无形和scala版本上工作,所以我怀疑我做错了什么.

https://github.com/milessabin/shapeless/wiki/Migration-guide:-shapeless-1.2.4-to-2.0.0#iso-is-now-generic

作为参考,我正在运行scala 2.11.6和无形2.2.2

Tra*_*own 5

你很近.问题是Scala不会自动为您调用隐式需求.如果你需要一个Generic[A, T]实例来调用convert,那么每次调用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))
Run Code Online (Sandbox Code Playgroud)

在这一变化之后,一切都应该正常工作.

请注意,您还可以通过添加如下所示的(抽象)方法来要求特征级别的实例:

implicit def genA: Generic.Aux[A, T]
Run Code Online (Sandbox Code Playgroud)

然后任何实现的类LoadsCsv都可以有一个隐式val genA参数(或者可以以其他方式提供实例).