Scala错误:缺少扩展函数的参数类型

时间:2015-06-19 14:14:34

标签: scala generics compiler-errors

我正在尝试为Scala编写查询库。这是迄今为止的代码:

class Query[TElement](source: Traversable[TElement]) {
    def join[TOther](other: Traversable[TOther]) = new {
        def on[TKey](keySelector1: TElement => TKey) = new {
            def equals(keySelector2: TOther => TKey) = new {
                def into[TResult](resultSelector: (TElement, TOther) => TResult): Query[TResult] = {
                    val map = source.map(e => (keySelector1(e), e)).toMap
                    val results = other
                        .map(e => (keySelector2(e), e))
                        .filter(p => map.contains(p._1))
                        .map(p => (map(p._1), p._2))
                        .map(p => resultSelector(p._1, p._2))
                    new Query[TResult](results)
                }
            }
        }
    }
}

object Query {
    def from[TElement](source: Traversable[TElement]): Query[TElement] = {
        new Query[TElement](source)
    }
}

...

val results = Query.from(users)
    .join(accounts).on(_.userId).equals(_.ownerUserId).into((_, _))

我去编译时遇到以下错误:

error: missing parameter type for expanded function ((x$2) => x$2.ownerUserId)

我有点困惑为什么我会在非泛型函数equals上出现此错误。我认为它的通用参数来自外部范围。我知道要修复它我必须通过编写(a: Account) => a.ownerUserId明确说明参数类型是什么。但是,我试图让它成为一个非常流畅的库,这使它变得混乱。

1 个答案:

答案 0 :(得分:2)

问题很简单。使用equals继承的现有方法Any存在歧义。简单的例子:

scala> class X[A, B] { def equals(f: A => B) = f }
defined class X

scala> val x = new X[Int, String]
x: X[Int,String] = X@52d455b8

scala> x.equals((x: Int) => x.toString)
res0: Int => String = <function1>

scala> x.equals((x: String) => x.toString) // uh-oh
res1: Boolean = false

正如在上一个示例中所看到的,当传递了错误的函数类型时,编译器必须选择def equals(Any): Boolean。如果您没有指定任何类型,编译器必须推断出一个,在示例中它不能做。

只需将您的方法重命名为其他内容,问题就不复存在了。