3

我正在查看页面Dotty下的文档Contextual Abstractions,我看到了Given Instances.

给定的实例(或者简单地说,“给定的”)定义了某些类型的“规范”值,这些值用于综合给定子句的参数。例子:

trait Ord[T] {
  def compare(x: T, y: T): Int
  def (x: T) < (y: T) = compare(x, y) < 0
  def (x: T) > (y: T) = compare(x, y) > 0
}

given intOrd: Ord[Int] {
  def compare(x: Int, y: Int) =
    if (x < y) -1 else if (x > y) +1 else 0
}

given listOrd[T]: (ord: Ord[T]) => Ord[List[T]] {

  def compare(xs: List[T], ys: List[T]): Int = (xs, ys) match {
    case (Nil, Nil) => 0
    case (Nil, _) => -1
    case (_, Nil) => +1
    case (x :: xs1, y :: ys1) =>
      val fst = ord.compare(x, y)
      if (fst != 0) fst else compare(xs1, ys1)
  }
}

但是文档中的这个示例从未解释过如何使用given. 我拉了测试Dotty示例项目并尝试使用它,但我不太了解它。

它是一个新的关键字吗?我们进口它吗?还是我错过了什么。

4

3 回答 3

7

这是使用given实例的示例。假设我们要比较两个整数,看看哪个比另一个大。我们可以利用intOrd上面已经定义的内容并编写:

def whichIsBigger[T](x: T, y: T)(given ord: Ord[T]): String = {
  ord.compare(x, y) match {
    case -1 => s"$x is less than $y"
    case 0 => s"$x and $y are equal"
    case 1 => s"$x is greater than $y"
  }
}

println(whichIsBigger(2, 1))

产生:

2 is greater than 1

我们能够这样做是因为在作用域中有一个命名的给定实例,否则,编译器会抱怨它没有Ord[Int].

它是一个新的关键字吗?我们进口它吗?或者我错过了什么。

它是一个新关键字,用于替换 Scala 2 中implicit定义的特定部分。如果这是 Scala 2,我们会这样写:

implicit val intOrd: Ord[Int] = new Ord[Int] {
  def compare(x: Int, y: Int) =
    if (x < y) -1 else if (x > y) 1 else 0
}

def whichIsBigger[T](x: T, y: T)(implicit ord: Ord[T]): String
于 2019-12-23T15:20:04.883 回答
4

implicit比较我们如何在 Scala 2 中使用关键字和在 Scala 3 中使用关键字来定义类型类可能会很有启发性given

斯卡拉 2

trait Semigroup[A] {
  def combine(x: A, y: A): A
}

object Semigroup {
  def combine[A: Semigroup](x: A, y: A) = implicitly[Semigroup[A]].combine(x,y)

  implicit val intSemigroup: Semigroup[Int] = new Semigroup[Int] {
    def combine(x: Int, y: Int) = x + y
  }

  implicit val quxSemigroup: Semigroup[Qux] = new Semigroup[Qux] {
    def combine(x: Qux, y: Qux) = Qux(x.a + y.a)
  }
}

case class Qux(a: Int)

Semigroup.combine(41, 1)
Semigroup.combine(Qux(41), Qux(1))

斯卡拉 3

trait Semigroup[A] {
  def combine(x: A, y: A): A
}

object Semigroup {
  def combine[A](x: A, y: A)(given Semigroup[A]) = summon.combine(x,y)

  given intSemigroup: Semigroup[Int] {
    def combine(x: Int, y: Int) = x + y
  }

  given quxSemigroup: Semigroup[Qux] {
    def combine(x: Qux, y: Qux) = Qux(x.a + y.a)
  }
}

case class Qux(a: Int)

Semigroup.combine(41, 1))
Semigroup.combine(Qux(41), Qux(1))
于 2019-12-23T15:26:34.587 回答
3

是的,它是一个新关键字,正如您可以从'given'页面末尾的语法中使用的那样(“语法”部分)看出这一点。它旨在取代implicit. 如果您已经熟悉隐式,我认为与 Scala 2 隐式的关系是很好的开始。

于 2019-12-23T15:22:10.610 回答