答案在 的定义中找到map
:
def map[B, That](f : (A) => B)(implicit bf : CanBuildFrom[Repr, B, That]) : That
请注意,它有两个参数。第一个是你的函数,第二个是隐式的。如果你不提供隐含的,Scala 将选择最具体的可用。
关于breakOut
那么,这样做的目的是breakOut
什么?考虑为问题给出的示例,您获取一个字符串列表,将每个字符串转换为一个 tuple (Int, String)
,然后从中产生一个Map
。最明显的方法是生成一个中间List[(Int, String)]
集合,然后转换它。
鉴于map
使用 aBuilder
来生成结果集合,是否可以跳过中介List
并将结果直接收集到 a 中Map
?显然,是的,是的。然而,要这样做,我们需要传递一个正确的CanBuildFrom
to map
,而这正是这样breakOut
做的。
那么,让我们看一下 的定义breakOut
:
def breakOut[From, T, To](implicit b : CanBuildFrom[Nothing, T, To]) =
new CanBuildFrom[From, T, To] {
def apply(from: From) = b.apply() ; def apply() = b.apply()
}
请注意,它breakOut
是参数化的,并且它返回CanBuildFrom
. 碰巧,类型From
和T
已经To
被推断出来,因为我们知道这map
是期待的CanBuildFrom[List[String], (Int, String), Map[Int, String]]
。所以:
From = List[String]
T = (Int, String)
To = Map[Int, String]
最后,让我们检查一下自己收到的隐式breakOut
。它是类型CanBuildFrom[Nothing,T,To]
。我们已经知道所有这些类型,所以我们可以确定我们需要一个隐式类型CanBuildFrom[Nothing,(Int,String),Map[Int,String]]
。但是有这样的定义吗?
我们看一下CanBuildFrom
的定义:
trait CanBuildFrom[-From, -Elem, +To]
extends AnyRef
所以CanBuildFrom
它的第一个类型参数是逆变的。因为Nothing
是一个底层类(即它是一切的子类),这意味着任何类都可以用来代替Nothing
.
由于存在这样的构建器,Scala 可以使用它来生成所需的输出。
关于建筑商
Scala 集合库中的许多方法包括获取原始集合,以某种方式对其进行处理(在 的情况下map
,转换每个元素),并将结果存储在新集合中。
为了最大限度地重用代码,这种结果的存储是通过构建器( scala.collection.mutable.Builder
) 完成的,它基本上支持两种操作:添加元素和返回结果集合。此结果集合的类型将取决于构建器的类型。因此,List
构建器将返回 a List
,Map
构建器将返回 a Map
,依此类推。方法的实现map
不需要关心结果的类型:构建器会处理它。
另一方面,这意味着map
需要以某种方式接收此构建器。设计 Scala 2.8 Collections 时面临的问题是如何选择最好的构建器。例如,如果我要写Map('a' -> 1).map(_.swap)
,我想得到一个Map(1 -> 'a')
回报。另一方面, aMap('a' -> 1).map(_._1)
不能返回 a Map
(它返回 a Iterable
)。
通过这种隐式Builder
执行从已知类型的表达式中产生最佳可能的魔力。CanBuildFrom
关于CanBuildFrom
为了更好地解释发生了什么,我将给出一个示例,其中被映射的集合是 aMap
而不是 a List
。我稍后再回去List
。现在,考虑以下两个表达式:
Map(1 -> "one", 2 -> "two") map Function.tupled(_ -> _.length)
Map(1 -> "one", 2 -> "two") map (_._2)
第一个返回 a Map
,第二个返回一个Iterable
。归还合身系列的魔力在于CanBuildFrom
. 让我们再考虑一下的定义map
来理解它。
该方法map
继承自TraversableLike
. 它在 and 上进行参数化B
,That
并使用类型参数A
andRepr
来参数化类。让我们一起看看这两个定义:
该类TraversableLike
定义为:
trait TraversableLike[+A, +Repr]
extends HasNewBuilder[A, Repr] with AnyRef
def map[B, That](f : (A) => B)(implicit bf : CanBuildFrom[Repr, B, That]) : That
要了解从何A
而来Repr
,让我们考虑一下Map
自身的定义:
trait Map[A, +B]
extends Iterable[(A, B)] with Map[A, B] with MapLike[A, B, Map[A, B]]
因为TraversableLike
被所有扩展的特征继承Map
,A
并且Repr
可以从它们中的任何一个继承。不过,最后一个获得了偏好。因此,根据不可变的定义以及Map
将其连接到 的所有特征TraversableLike
,我们有:
trait Map[A, +B]
extends Iterable[(A, B)] with Map[A, B] with MapLike[A, B, Map[A, B]]
trait MapLike[A, +B, +This <: MapLike[A, B, This] with Map[A, B]]
extends MapLike[A, B, This]
trait MapLike[A, +B, +This <: MapLike[A, B, This] with Map[A, B]]
extends PartialFunction[A, B] with IterableLike[(A, B), This] with Subtractable[A, This]
trait IterableLike[+A, +Repr]
extends Equals with TraversableLike[A, Repr]
trait TraversableLike[+A, +Repr]
extends HasNewBuilder[A, Repr] with AnyRef
如果您将类型参数Map[Int, String]
一路向下传递,我们会发现传递给 的类型TraversableLike
,并因此被 使用map
,是:
A = (Int,String)
Repr = Map[Int, String]
回到这个例子,第一个 map 接收一个 type 的函数,((Int, String)) => (Int, Int)
第二个 map 接收一个 type 的函数((Int, String)) => String
。我使用双括号来强调它是一个正在接收的元组,因为这是A
我们看到的类型。
有了这些信息,让我们考虑其他类型。
map Function.tupled(_ -> _.length):
B = (Int, Int)
map (_._2):
B = String
我们可以看到第一个返回的类型map
是Map[Int,Int]
,第二个是Iterable[String]
。查看map
的定义,很容易看出这些是 的值That
。但它们来自哪里?
如果我们查看所涉及类的伴生对象,我们会看到一些提供它们的隐式声明。在对象上Map
:
implicit def canBuildFrom [A, B] : CanBuildFrom[Map, (A, B), Map[A, B]]
在 object 上Iterable
,其类通过以下方式扩展Map
:
implicit def canBuildFrom [A] : CanBuildFrom[Iterable, A, Iterable[A]]
这些定义为参数化提供了工厂CanBuildFrom
。
Scala 将选择最具体的可用隐式。在第一种情况下,它是第一个CanBuildFrom
。在第二种情况下,由于第一种不匹配,它选择了第二种CanBuildFrom
。
回到问题
让我们看看问题的代码,List
's 和map
' 的定义(再次)看看类型是如何推断的:
val map : Map[Int,String] = List("London", "Paris").map(x => (x.length, x))(breakOut)
sealed abstract class List[+A]
extends LinearSeq[A] with Product with GenericTraversableTemplate[A, List] with LinearSeqLike[A, List[A]]
trait LinearSeqLike[+A, +Repr <: LinearSeqLike[A, Repr]]
extends SeqLike[A, Repr]
trait SeqLike[+A, +Repr]
extends IterableLike[A, Repr]
trait IterableLike[+A, +Repr]
extends Equals with TraversableLike[A, Repr]
trait TraversableLike[+A, +Repr]
extends HasNewBuilder[A, Repr] with AnyRef
def map[B, That](f : (A) => B)(implicit bf : CanBuildFrom[Repr, B, That]) : That
的类型List("London", "Paris")
是List[String]
,所以类型A
和Repr
定义TraversableLike
为:
A = String
Repr = List[String]
的类型(x => (x.length, x))
是(String) => (Int, String)
,所以 的类型B
是:
B = (Int, String)
最后一个未知类型That
是 的结果的类型map
,我们也已经有了:
val map : Map[Int,String] =
所以,
That = Map[Int, String]
这意味着breakOut
必须返回一个类型或子类型CanBuildFrom[List[String], (Int, String), Map[Int, String]]
。