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

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.

主站蜘蛛池模板: 万山特区| 元氏县| 广安市| 云安县| 千阳县| 漳平市| 剑川县| 额济纳旗| 武穴市| 江口县| 措勤县| 永平县| 南漳县| 景洪市| 额敏县| 米脂县| 平遥县| 陵水| 营口市| 卓资县| 龙游县| 张家口市| 峡江县| 许昌县| 云和县| 大名县| 三都| 惠来县| 新源县| 新蔡县| 布尔津县| 铜川市| 四平市| 扶沟县| 康定县| 永宁县| 呼和浩特市| 宜黄县| 遂溪县| 峨山| 夏河县|