7

假设我有一个类型类Graph[G,V],它声明一个类型的对象G也是一个具有类型顶点的图V

现在我有一个隐含的方法,可以让我将类型对的集合A视为具有类型顶点的图A(无法表示未连接的顶点......)。我可以通过导入以下对象的范围来使用隐式。

object TupleSetGraph{
  implicit def ts2graph[A]: Graph[Set[(A,A)],A] = new Graph[Set[(A,A)],A] {
    def nodes(g: Set[(A, A)]): Set[A] = g flatMap (t => Set(t._1,t._2))
    def adjacent(g: Set[(A, A)], n1: A, n2: A): Boolean = g.contains((n1,n2)) || g.contains((n2,n1))
  }
}

假设我还希望能够映射顶点的内容,从而能够执行以下操作:

(_: Set[(A,A)]).map((_: A => B)): Set[(B,B)]

但是已经有一个map定义了Set。如何处理相同的数据结构可以map以不同的方式被视为相同的东西(具有功能的东西)的问题?

4

3 回答 3

3

绘制一个可能的解决方案:

将地图操作放在辅助特征中

GraphOps(这可能是Graph它本身,但地图签名可能太复杂了)

case class GraphOps[G](data: G) { def map...}

轻松获得GraphOps

object Graph {
   def apply[G](data: G) = GraphOps(data)
}

这样,电话将是

Graph(set).map(f) 

apply可以隐含,但我不确定我想这样做(如果我这样做了,我不确定它会正确找到地图)。

变体。在 GraphOps 中有图表

我们也可以

case class GraphOps[G,V](data: G, graph: Graph[G,V])

object Graph {
   def apply[G,V](data: G)(implicit graph: Graph[G,V]) = GraphOps(data, graph)
}

这样做的好处是顶点类型V在 GraphOps 中可用

定义地图操作

您想要的签名很复杂,Set[(A,A)] 返回一个 Set[(B,B)],但其他图形实现返回完全不同的东西。这类似于在集合库中所做的。

我们可以引入一个特征 CanMapGraph[From, Elem, To],类似于 CanBuildFrom

trait CanMapGrap[FromGraph, FromElem, ToGraph, ToElem] {
  def map(data: FromGraph, f: FromElem => ToElem): ToGraph
}

(您可能会将其更改为具有比 map 更多的基本操作,以便可以将其用于不同的操作,如使用 完成CanBuildFrom

然后地图将是

case class GraphOps[G](data: G) {
  def map[A,B](f: A, B)(implicit ev: CanMapFrom[G, A, B, G2]) : G2 =
    ev.map(data, f)
}

你可以定义

implicit def mapPairSetToPairSet[A, B] = 
  new CanMapGraph[Set[(A,A)], A, Set[(B,B)], B] {
    def map(set: Set[(A,A)], f: A => B) = set.map{case (x, y) => (f(x), f(y))}
  } 

然后你做

val theGraph = Set("A" -> "B", "BB" -> "A", "B" -> "C", "C" -> "A")
Graph(theGraph).map(s: String -> s(0).toLower)
res1: Set[(Char, Char)] = Set((a,b), (b,a), (b,c), (c,a))

一个问题是顶点的类型在第一个参数列表中是未知的,即 f 的那个,所以我们必须明确地使用 s: String。

使用另一种方法GraphOps,我们提前获取顶点类型,A不是 Map 的参数,而是 的参数GraphOps,因此从一开始就知道,不需要在 中显式f。如果您这样做,您可能希望将图形传递给map.CanMapGraph

使用第一个解决方案,仍然很容易将图形提供给CanMapGraph.

implicit def anyGraphToSet[G,V,W](implicit graph: Graph[G,V]) 
  = new CanMapFrom[G, V, Set[(W,W)], W] {
    def map(data: G, f: V => W) = 
      (for {
         from <- graph.nodes(data)
         to <- graph.nodes(data)) 
         if graph.adjacent(data, from, to) }
       yield (from, to)).toSet
  }
于 2011-10-25T16:11:39.253 回答
1
val x: Set[(A, A)] = ...
(x: Graph[_, _]).map(...)

如果您希望名称相同,这似乎是您能做的最好的事情。

正如您所指出的,这不是您想要的。这应该会更好:

object Graph {
  def map[G, V](graph: G)(f: V => V)(implicit instance: Graph[G, V]) = ...
}

val x: Set[(A, A)] = ...
Graph.map(x)(f) 
// but note that the type of argument of f will often need to be explicit, because
// type inference only goes from left to right, and implicit arguments come last

请注意,您只能让fto beV => V和 not V => V1。为什么?想象一下,你有 implicit g1: Graph[SomeType, Int],但没有implicit g2: Graph[SomeType, String]。那还能Graph.map(_: SomeType)((_: Int).toString)回来什么?这个问题可以通过要求G参数化类型来避免:

trait Graph[G[_]] {
  def nodes[A](g: G[A]): Set[A]
  def adjacent[A](g: G[A], n1: A, n2: A): Boolean
}

object TupleSetGraph{
  type SetOfPairs[A] = Set[(A,A)]
  implicit def ts2graph: Graph[SetOfPairs] = new Graph[SetOfPairs] {
    def nodes[A](g: Set[(A, A)]): Set[A] = g flatMap (t => Set(t._1,t._2))
    def adjacent[A](g: Set[(A, A)], n1: A, n2: A): Boolean = g.contains((n1,n2)) || g.contains((n2,n1))
  }
}

那么你有

object Graph {
  def map[G[_], V, V1](graph: G[V])(f: V => V1)(implicit instance: Graph[G]) = ...
}
于 2011-10-25T13:21:01.330 回答
0

如果您使用的是类型类,那么您可以执行以下操作:

implicitly[TypeClass].map(...)

如果您使用的是视图边界,那么 Alexey 的回答是正确的:

(...: ViewBound).map(...)
于 2011-10-25T16:55:59.840 回答