Вы можете, довольно круглым способом. Foo
является классом типа, и компилятор неявно передает экземпляр класса типа, совместимый с (выводимым) параметром типа A
.
trait Foo[X] {
def apply(xs: Seq[X]): Unit
}
object Foo {
implicit def FooAny[A]: Foo[A] = new Foo[A] {
def apply(xs: Seq[A]) = println("apply(xs: Seq[A])")
}
implicit def FooTuple2[A, B]: Foo[(A, B)] = new Foo[(A, B)] {
def apply(xs: Seq[(A, B)]) = println("apply(xs: Seq[(A, B)])")
}
def apply[A](xs: A*)(implicit f: Foo[A]) = f(xs)
}
Foo(1, 2, 3) // apply(xs: Seq[A])
Foo(1 -> 2, 2 -> 3) // apply(xs: Seq[(A, B)])
Во втором вызове можно передать как FooAny
, так и FooTuple2
, но компилятор выбирает FooTuple2
, основываясь на правилах перегрузки статического метода. FooTuple2
считается более конкретным, чем FooAny
. Если два кандидата считаются такими же конкретными, как и каждый другой, возникает ошибка неоднозначности. Вы также можете отдать предпочтение одному над другим, поместив один в суперкласс, как это сделано в scala.LowPriorityImplicits
.
UPDATE
Отказ от идеи DummyImplicit и последующего потока на scala-user:
trait __[+_]
object __ {
implicit object __ extends __[Any]
}
object overload {
def foo(a: Seq[Boolean]) = 0
def foo[_: __](a: Seq[Int]) = 1
def foo[_: __ : __](a: Seq[String]) = 2
}
import overload._
foo(Seq(true))
foo(Seq(1))
foo(Seq("s"))
Здесь объявляется параметризованная черта типа __
, ковариантная в параметре безымянного типа _
. Его сопутствующий объект __
содержит неявный экземпляр __[Any]
, который нам понадобится позже. Вторая и третья перегрузки foo
включают параметры фиктивного типа, опять же безымянные. Это будет выведено как Any
. Этот параметр типа имеет одну или несколько границ контекста, которые разбиты на дополнительные неявные параметры, например:
def foo[A](a: Seq[Int])(implicit ev$1: __[A]) = 1
Несколько списков параметров объединяются в один список параметров в байт-коде, поэтому проблема двойного определения обходится.
Пожалуйста, рассмотрите это как возможность узнать о стирании, контекстных границах и неявном поиске, а не как шаблон, который будет применяться в реальном коде!