Возможно ли "каррировать" более высокие типы в Scala?

Предположим, что у меня есть черта с двумя параметрами типа, например

trait Qux[A, B]

и другой признак с параметром более высокого типа, например

trait Turkle[C[_]]

Я хотел бы иметь возможность подставить фиксированное значение для одного из параметров типа для Qux, чтобы его можно было использовать для параметризации Turkle.

Вот пример (кода, который не имеет смысла в Scala!):

trait Baz[A] extends Turkle[Qux[A, _]]

У кого-нибудь есть идеи, как добиться этого эффекта?

Ответы

Ответ 1

Джейсон Заугг придумал самый лаконичный способ сделать это:

trait Baz[A] extends Turkle[({type x[a]=Qux[A, a]})#x]

Плагин IntelliJ Scala по желанию может свернуть его, чтобы:

trait Baz[A] extends Turkle[x[a]=Qux[A, a]]

Ответ 2

Вы имеете в виду что-то вроде этого?

trait QuxWithString[A] extends Qux[A, String]
new Turkle[QuxWithString]{}

Это аналоговое частичное приложение для типов.

Ответ 3

trait Turkle[C[_]]
trait Qux[A,B]
trait Wraps[A] {
  type Jkz[X] = Qux[A,X]
  trait Baz extends Turkle[Jkz]
}

Ответ 4

Плагин компилятора добрый проектор также позволяет:

// Explicit lambda, greek letters
trait Baz[A] extends Turkle[λ[α=>Qux[A,α]]]

// Explicit lambda, normal letters
trait Baz[A] extends Turkle[Lambda[a=>Qux[A,a]]]

// No explicit lambda, ? placeholder    
trait Baz[A] extends Turkle[Qux[A,?]]