2

下面是我有时在使用参数化类型时遇到的一种情况的具体实例。基本上,我知道有些类型参数是兼容的,但我不知道如何向代码的某些部分证明这一点。

我正在编写一个将 url 映射到处理函数的请求路由器。下面是一些简化的代码。我创建了一个List[Route],其中 aRoute基本上是UrlMatcher, Function一对。

class Route[A](matcher: UrlMatcher[A], handler: HandlerFunction[A])

abstract class UrlMatcher[A] {
   def match(url: String): Option[A]   // None if no match

type 参数A用于匹配器可能从 URL 中提取的“参数”。它们将被传递给处理函数。例如,UrlMatcher[Int]看到像“/users/123”这样的 URL 路径的 a 可以将 123 传递给getUser(id: Int)函数。路由器可能如下所示:

val routes = ArrayBuffer[Route[_]]

def callHandler(url: String) {
  for (r <- routes) {
    val args = r.matcher.matchUrl(url)
    if (args.isDefined)
      r.handler(args.get)  // <--- error here
    }

问题是我得到类型不匹配错误,因为我不知道如何告诉它这两种类型是相同的。

type mismatch; found: args.type (with underlying type Option[Any])  
            required: _$1  

我知道我可以重新设计它,使其Route具有类似的方法matchAndCall,但如果可能的话,我想保持这种逻辑流程。

更新/编辑

我不完全理解存在类型,但我试过这个......

val routes = ArrayBuffer[T forSome { type T }]()

它消除了上面的不匹配错误。但是,我还有另一个插入到ArrayBuffer.

def route[P](matcher: UrlMatcher[P], handler: Handler[P]): AbstractRoute = {
  val route = new Route(matcher, handler)
  otherRoutes.append(route)   // error here  
  route
}

现在错误是...

type mismatch;  found : Route[P]  required: Route[T forSome { type T }] Note: P <: T
forSome { type T }, but class Route is invariant in type P. You may wish to define 
P as +P instead. (SLS 4.5) 

为什么P不兼容T,既然对它们没有约束T

4

1 回答 1

3

这是存在类型(通配符类型的 Scala 等价物)是坏事 (TM) 的原因之一,在不进行 Java 互操作时最好避免这种情况:编译器不能(或不够聪明)通常推理哪个types 等于 which,因为它们都消失了……

为了使编译器理解这些类型是相同的,您需要以某种方式为该类型命名。

类型参数是一种可能性:您使用 for 理解的内容定义参数化方法,以便在方法中,类型是众所周知的。

def callHandler(url: String) {
  def call[T](r: Route[T]) = {
    val args = r.matcher.matchUrl(url)
    if (args.isDefined) r.handler(args.get)
    // or args.foreach(r.handler)
  }
  for (r <- routes) call(r)
  // or routes.foreach(call)
}

注意:在更简单的情况下,您也可以使用方差来拥有 a List[Route[Any]],并且您的问题消失了,该类型又是众所周知的。在这里,我不确定您是否可以使Route[A]协变。

存在类型主要用于表示 Java 通配符、Java 原始类型和 JVM 的类型视图(反射和东西),尽管它们比这三种结构更强大如果你可以设计一些东西来避免使用它们,你会为自己省去很多痛苦。这有点争议,但至少,它们与类型推断交互的方式有很多限制,所以必须小心。

于 2013-09-13T17:56:48.997 回答