在Android中使用Kotlin擴展

在 Java 中如果我們要為類添加新功能枉昏,就必須使用繼承或者像裝飾者這樣的設(shè)計模式萄凤,但是在 Kotlin 中這些可以通過叫做擴展的方式來完成黍特。平時我們開發(fā) Android 的過程中,會逐漸總結(jié)出各種各樣的工具類嫉鲸,如果使用 Kotlin 則可以通過擴展的方式产上,來簡化或者替代這些工具類棵磷。

Kotlin 擴展包括擴展屬性、擴展函數(shù) 2 種方式蒂秘,具體語法這里不多說了泽本,主要講講通過擴展,在 Android 開發(fā)中可以為我們帶來哪些便利姻僧。

1. 替代View.setOnClickListener方法

在 Java 中為一個 View 設(shè)置點擊事件,我們必須這樣寫:

view.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View v) {
    }
});

如果一個頁面中要為很多 View 設(shè)置點擊事件蒲牧,寫起來巨繁瑣撇贺,在 Kotlin 中我們可以為 View 擴展一個函數(shù)來輕松實現(xiàn)點擊事件:

fun <T : View> T.click(block: (T) -> Unit) {
    setOnClickListener {
        block(this)
    }
}

這個時候,我們?yōu)?View 設(shè)置點擊事件可以這樣寫:

view.click {
   //處理點擊事件邏輯
}

在寫法上是不是簡單多了冰抢,除此之外還可通過擴展屬性來實現(xiàn)更高級功能松嘶。舉個栗子,我們不希望某個按鈕被頻繁點擊挎扰,在 Java 中的做法是設(shè)置一個變量記錄點擊時間翠订,如果在某個時間間隔內(nèi)再次觸發(fā)點擊巢音,則不響應(yīng)此事件。使用 Kotlin 我們可以這樣實現(xiàn)該功能:

先為 View 擴展 2 個屬性:

//私有擴展屬性尽超,允許2次點擊的間隔時間
private var <T : View> T.delayTime: Long
    get() = getTag(0x7FFF0001) as? Long ?: 0
    set(value) {
        setTag(0x7FFF0001, value)
    }

//私有擴展屬性官撼,記錄點擊時的時間戳
private var <T : View> T.lastClickTime: Long
    get() = getTag(0x7FFF0002) as? Long ?: 0
    set(value) {
        setTag(0x7FFF0002, value)
    }   

再為 View 擴展一個方法:

//私有擴展方法,判斷能否觸發(fā)點擊事件
private fun <T : View> T.canClick(): Boolean {
    var flag = false
    var now = System.currentTimeMillis()
    if (now - this.lastClickTime >= this.delayTime) {
        flag = true
        this.lastClickTime = now
    }
    return flag
}

//擴展點擊事件似谁,默認 500ms 內(nèi)不能觸發(fā) 2 次點擊
fun <T : View> T.clickWithDuration(time: Long = 500, block: (T) -> Unit) {
    delayTime = time
    setOnClickListener {
        if (canClick()) {
            block(this)
        }
    }
}

在 Kotlin 中使用傲绣,我們只需這樣調(diào)用:

//只有間隔1秒之后,點擊才會生效
view.clickWithDuration(1000) {
    //點擊事件
}

上面這個例子中巩踏,實際上是通過 setTag()/getTag() 來存儲 2 個擴展屬性真正的值的秃诵。這是因為擴展屬性并沒有將實際的成員插入類中,他們只能由顯示提供的 getters/setters 定義塞琼。

2. Context的一些擴展

在 Java 中跳轉(zhuǎn)到另一個頁面菠净,一般得這樣寫:

Intent intent = new Intent(context, MainActivity.class);
context.startActivity(intent)

使用 Kotlin 后我們可以這樣寫:

//使用內(nèi)聯(lián)函數(shù)的泛型參數(shù) reified 特性來實現(xiàn)
inline fun <reified T : Activity> Context.startActivity() {
    val intent = Intent(this, T::class.java)
    if (this !is Activity) {
        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
    }
    startActivity(intent)
}

//這個時候跳轉(zhuǎn)可以這樣寫,是不是特別簡潔
Context.startActivity<MainActivity>()

還有很多我們常用的方法:

//屏幕寬度(px)
inline val Context.screenWidth: Int
    get() = resources.displayMetrics.widthPixels
    
//屏幕高度(px)
inline val Context.screenHeight: Int
    get() = resources.displayMetrics.heightPixels

//屏幕的密度
inline val Context.density: Float
    get() = resources.displayMetrics.density    
    
//dp 轉(zhuǎn)為 px
inline fun Context.dp2px(value: Int): Int = (density * value).toInt()

//dp 轉(zhuǎn)為 px
inline fun Context.dp2px(value: Float): Int = (density * value).toInt()

//px 轉(zhuǎn)為 dp
inline fun Context.px2dp(value: Int): Float = value.toFloat() / density

在實際使用時彪杉,這些方法和屬性就像類本身定義的一樣毅往,我們可以直接拿來使用,非常方便在讶,省去了類似的工具類方法調(diào)用煞抬。

3. 可空接收者

在 Java 中字符串操作很容易出現(xiàn)空指針異常,要判斷一個字符串為空构哺,一般得這樣判斷:

String str = null;

//對字符串 str 做是否為空判斷
if (str == null || str.length() == 0) {

}

在 Kotlin 中我們要判斷字符串是否為空革答,sdk 里直接提供了一個方法 isNullOrEmpty,看源碼是通過擴展來實現(xiàn)的曙强,主要代碼如下:

public inline fun CharSequence?.isNullOrEmpty(): Boolean {   
    return this == null || this.length == 0
}

該方法與前面擴展函數(shù)在形式上的唯一差別是残拐,在類型后面跟著的是?.而不是.,然后在方法體內(nèi)可以通過this == null來判斷調(diào)用該方法的對象是否為 null碟嘴,上面的 Java 代碼我們用 Kotlin 來寫:

var str = null
//這里再也不需要額外對 str 做非空判斷了溪食,可以放心的使用了
if (str.isNullOrEmpty()) {

}

使用這種方式來擴展函數(shù),媽媽再也不用擔心空指針異常了娜扇。

4. 小結(jié)

在 Kotlin sdk 里错沃,可以發(fā)現(xiàn)它采用了大量的擴展函數(shù)。合理使用擴展雀瓢,可以大量簡化代碼枢析,提升開發(fā)效率,這個是 Kotlin 中我最喜歡的特性之一刃麸。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末醒叁,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌把沼,老刑警劉巖啊易,帶你破解...
    沈念sama閱讀 211,348評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異饮睬,居然都是意外死亡租谈,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,122評論 2 385
  • 文/潘曉璐 我一進店門续捂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來垦垂,“玉大人,你說我怎么就攤上這事牙瓢〗俎郑” “怎么了?”我有些...
    開封第一講書人閱讀 156,936評論 0 347
  • 文/不壞的土叔 我叫張陵矾克,是天一觀的道長页慷。 經(jīng)常有香客問我,道長胁附,這世上最難降的妖魔是什么酒繁? 我笑而不...
    開封第一講書人閱讀 56,427評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮控妻,結(jié)果婚禮上州袒,老公的妹妹穿的比我還像新娘。我一直安慰自己弓候,他們只是感情好郎哭,可當我...
    茶點故事閱讀 65,467評論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著菇存,像睡著了一般夸研。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上依鸥,一...
    開封第一講書人閱讀 49,785評論 1 290
  • 那天亥至,我揣著相機與錄音,去河邊找鬼贱迟。 笑死姐扮,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的衣吠。 我是一名探鬼主播溶握,決...
    沈念sama閱讀 38,931評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼蒸播!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,696評論 0 266
  • 序言:老撾萬榮一對情侶失蹤袍榆,失蹤者是張志新(化名)和其女友劉穎胀屿,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體包雀,經(jīng)...
    沈念sama閱讀 44,141評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡宿崭,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,483評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了才写。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片葡兑。...
    茶點故事閱讀 38,625評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖赞草,靈堂內(nèi)的尸體忽然破棺而出讹堤,到底是詐尸還是另有隱情,我是刑警寧澤厨疙,帶...
    沈念sama閱讀 34,291評論 4 329
  • 正文 年R本政府宣布洲守,位于F島的核電站,受9級特大地震影響沾凄,放射性物質(zhì)發(fā)生泄漏梗醇。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,892評論 3 312
  • 文/蒙蒙 一撒蟀、第九天 我趴在偏房一處隱蔽的房頂上張望叙谨。 院中可真熱鬧,春花似錦保屯、人聲如沸手负。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,741評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽虫溜。三九已至,卻和暖如春股缸,著一層夾襖步出監(jiān)牢的瞬間衡楞,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評論 1 265
  • 我被黑心中介騙來泰國打工敦姻, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留瘾境,地道東北人。 一個月前我還...
    沈念sama閱讀 46,324評論 2 360
  • 正文 我出身青樓镰惦,卻偏偏與公主長得像迷守,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子旺入,可洞房花燭夜當晚...
    茶點故事閱讀 43,492評論 2 348

推薦閱讀更多精彩內(nèi)容