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

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.

主站蜘蛛池模板: 南江县| 鄂托克前旗| 保靖县| 甘肃省| 华池县| 长沙市| 寿宁县| 香港| 新丰县| 南丹县| 吉隆县| 大余县| 阿图什市| 南雄市| 祥云县| 禹城市| 天祝| 德阳市| 连江县| 离岛区| 荥阳市| 独山县| 敦化市| 札达县| 中山市| 突泉县| 娱乐| 利川市| 蒙自县| 遵义市| 威远县| 华亭县| 略阳县| 城固县| 邻水| 宝清县| 丰原市| 吐鲁番市| 大关县| 荆门市| 林州市|