近日拜讀了大佬 mikyou 一篇講 Kotlin 中泛型的 文章,里面說到 Kotlin 中泛型應(yīng)用場(chǎng)景的問題時(shí)提到了給 Boolean 擴(kuò)展出一套流式 API 的點(diǎn)子,甚是精彩,舉個(gè)栗子:
(3 < 2).yes {
toast("yes")
}.otherwise {
toast("otherwise")
}
當(dāng)然上面的代碼還體現(xiàn)不出來給 Boolean 擴(kuò)展流式 API 的優(yōu)越性,我就是隨便舉個(gè)栗子給大家看看~
源碼如下:
sealed class BooleanExt<out T>//起橋梁作用的中間類,定義成協(xié)變
object Otherwise : BooleanExt<Nothing>()//Nothing是所有類型的子類型,協(xié)變的類繼承關(guān)系和泛型參數(shù)類型繼承關(guān)系一致
class TransferData<T>(val data: T) : BooleanExt<T>()//data只涉及到了只讀的操作
//聲明成inline函數(shù)
inline fun <T> Boolean.yes(block: () -> T): BooleanExt<T> = when {
this -> {
TransferData(block.invoke())
}
else -> Otherwise
}
inline fun <T> BooleanExt<T>.otherwise(block: () -> T): T = when (this) {
is Otherwise ->
block()
is TransferData ->
this.data
}
很騷的操作有木有吕世,不過我看后還是產(chǎn)生了點(diǎn)私人想法,這里調(diào)用這套流式 API
的時(shí)候梯投,必須先判斷肯定命辖,再判斷否定,我在想能不能再擴(kuò)展下可以做到先判斷否定分蓖,再判斷肯定也沒問題:
(3 < 2).otherwise {
toast("otherwise")
}.yes {
toast("yes")
}
當(dāng)然沒問題了尔艇!不過這得多定義個(gè)中間類,兩個(gè)中間類么鹤,最終定義在兩個(gè)文件里兼顧兩種調(diào)用順序漓帚。廢話不多說,我們直接貼代碼午磁,就是在項(xiàng)目里定義兩個(gè)文件尝抖,一個(gè) BooleanExt1,一個(gè) BooleanExt2:
/**
* Created by xiaofei on 2018/12/30.
* desc:Boolean Extension1, Say Goodbye to if-else expression
*/
sealed class BooleanExt1<out T>////起橋梁作用的中間類迅皇,定義成協(xié)變
object Otherwise : BooleanExt1<Nothing>()//Nothing是所有類型的子類型昧辽,協(xié)變的類繼承關(guān)系和泛型參數(shù)類型繼承關(guān)系一致
class TransferData1<T>(val data: T) : BooleanExt1<T>()//data只涉及到了只讀的操作
//聲明成inline函數(shù)
inline fun <T> Boolean.yes(block: () -> T): BooleanExt1<T> = when {
this -> {
TransferData1(block.invoke())
}
else -> Otherwise
}
inline fun <T> BooleanExt1<T>.otherwise(block: () -> T): T = when (this) {
is Otherwise ->
block()
is TransferData1 ->
this.data
}
and
/**
* Created by xiaofei on 2018/12/30.
* desc:Boolean Extension2, Say Goodbye to if-else expression
*/
sealed class BooleanExt2<out T>////起橋梁作用的中間類,定義成協(xié)變
object Yes : BooleanExt2<Nothing>()//Nothing是所有類型的子類型登颓,協(xié)變的類繼承關(guān)系和泛型參數(shù)類型繼承關(guān)系一致
class TransferData2<T>(val data: T) : BooleanExt2<T>()//data只涉及到了只讀的操作
//聲明成inline函數(shù)
inline fun <T> Boolean.otherwise(block: () -> T): BooleanExt2<T> = when {
!this -> {
TransferData2(block.invoke())
}
else -> Yes
}
inline fun <T> BooleanExt2<T>.yes(block: () -> T): T = when (this) {
is Yes ->
block()
is TransferData2 ->
this.data
}
這兩個(gè)文件定義好之后搅荞,使用起來就可以很隨意了:
(3 < 2).yes {
toast("yes")
}.otherwise {
toast("otherwise")
}
(3 <2 ).otherwise {
toast("otherwise")
}.yes {
toast("yes")
}
完!