Scala类型类隐式解析

Ale*_*rov 5 scala implicit typeclass

(斯卡拉2.11.8)

请考虑以下代码:

object ScalaTest extends App {
  class Wrapper {
    import Wrapper._

    def init(): Unit = {
      // "could not find implicit value for parameter tc: ScalaTest.Wrapper.TC[Int]"
      printWithTC(123)

      // Compiles
      printWithTC(123)(IntTC)

      // Compiles again!
      printWithTC(132)
    }
  }

  object Wrapper {
    trait TC[A] {
      def text(a: A): String
    }

    implicit object IntTC extends TC[Int] {
      override def text(a: Int) = s"int($a)"
    }

    def printWithTC[A](a: A)(implicit tc: TC[A]): Unit = {
      println(tc.text(a))
    }
  }

  (new Wrapper).init()
}
Run Code Online (Sandbox Code Playgroud)

关于这段代码,我有很多问题:

  1. 为什么不IntTC首先解决?
  2. 为什么它在使用一次后编译?(如果您注释掉第一次调用,代码可以正常工作)
  3. 应该在哪里设置类型隐含以便正确解决?

Jas*_*r-M 3

使用val具有显式返回类型的 a。请参阅https://github.com/scala/bug/issues/801https://github.com/scala/bug/issues/8697(等等)。
隐式对象与具有推断返回类型的隐式 val 和 def 具有相同的问题。至于你的第二个问题:当IntTC显式使用时,你会强制编译器对其进行类型检查,因此在那之后它的类型是已知的并且可以通过隐式搜索找到。

class Wrapper {
  import Wrapper._

  def init(): Unit = {
    // Compiles
    printWithTC(123)

    // Compiles
    printWithTC(123)(IntTC)

    // Compiles
    printWithTC(132)
  }
}

object Wrapper {
  trait TC[A] {
    def text(a: A): String
  }

  implicit val IntTC: TC[Int] = new TC[Int] {
    override def text(a: Int) = s"int($a)"
  }

  def printWithTC[A](a: A)(implicit tc: TC[A]): Unit = {
    println(tc.text(a))
  }
}
Run Code Online (Sandbox Code Playgroud)

如果您确实希望像对象一样延迟评估隐式类型,则可以使用implicit lazy val带有显式类型的类型。