官术网_书友最值得收藏!

Type lambdas

As a next step, let's imagine that we have a generic Filler that is capable of filling different containers with different kinds of contents, as shown in the following code snippet:

sealed trait Contents
case class Water(purity: Int) extends Contents
case class Whiskey(label: String) extends Contents

sealed trait Container[C] { def contents: C }
case class Glass[C](contents: C) extends Container[C]
case class Jar[C](contents: C) extends Container[C]

sealed trait Filler[C <: Contents, CC <: Container[C]] {
def fill(c: C): CC
}

What could we do if we had a requirement to provide a method that should only accept one type of container or content? We would need to fix the second type parameter in a similar fashion to how we would partially apply a function if given one of the arguments. A type alias can be used to do this on the type level:

type WaterFiller[CC <: Container[Water]] = Filler[Water, CC]

def fillWithWater[CC <: Container[Water]](container: CC)(filler: WaterFiller[CC]) = ???

But it feels a bit verbose to define a type alias just to be used once in the definition of the function parameter. Type lambda is a syntax that allows us to do such partial type application in-place:

def fillWithWater[CC <: Container[Water], F: ({ type T[C] = Filler[Water, C] })#T[CC]](container: CC)(filler: F) = ???

The type lambda can also be used to define a parameter type directly:

def fillWithWater[CC <: Container[Water]](container: CC)(filler: ({ type T[C] = Filler[Water, C] })#T) = ???

The internal definition of T[C] is analogous to the type alias we defined previously. The added part is the type projection, ()#T[C], that allows us to reference the type we've just defined.

主站蜘蛛池模板: 拉萨市| 和顺县| 寻乌县| 乐昌市| 资阳市| 会同县| 广水市| 靖安县| 雷山县| 莱西市| 宜春市| 响水县| 张家港市| 阜城县| 灵寿县| 聂拉木县| 诸城市| 望都县| 天水市| 苍溪县| 海林市| 特克斯县| 宽甸| 西华县| 射洪县| 赤峰市| 加查县| 双辽市| 桂平市| 福海县| 靖宇县| 浦城县| 彰化市| 黎平县| 日土县| 满城县| 建平县| 宁阳县| 潍坊市| 通江县| 乐安县|