如何利用Kotlin實現(xiàn)極簡回調(diào)
前言
在各種開發(fā)場景中,回調(diào)都有著廣泛的應(yīng)用,命名往往是各種Callback和Listener,其中在Android中接觸最早也最常用的可能就是View.OnClickListener了。
mBtn.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Log.d("MM","Click"); } });
不過寫多了也有點煩惱,我只想打印一條日志,卻寫了這么多代碼。不過好在這個接口里面只包含一個方法,但換做一些包含方法數(shù)量比較多的回調(diào)就顯得比較臃腫了:
mEdit.addTextChangedListener(new TextWatcher() { @Override public void beforeTextChanged(CharSequence s, int start, int count, int after) { } @Override public void onTextChanged(CharSequence s, int start, int before, int count) { } @Override public void afterTextChanged(Editable s) { } });
如果你想優(yōu)化你的代碼,讓它們看起來更簡潔優(yōu)雅,可以試試Kotlin的中的一些方法。
簡化
先來看下Kotlin中的回調(diào):
mBtn.setOnClickListener(object :View.OnClickListener{ override fun onClick(v: View?) { println("Click") } })
好像一點也沒簡化嘛,不過因為在 Kotlin 里函數(shù)也是參數(shù)的一種,在 Java 中只包含一個方法的接口,在 Kotlin 中都可以使用 Lambda 表達式來達成一樣的效果。
mBtnCallback.setOnClickListener { println("Click") }
是不是簡單很多了,但上面的用法僅適用于接口中只有一個方法的情況,如果存在多個方法的話,當然也可以簡化了:
mEdit.addTextChangedListener { beforeTextChanged { text, start, count, after -> println("beforeTextChanged") } onTextChanged { text, start, before, count -> println("onTextChanged") } afterTextChanged { text -> println("afterTextChanged") } }
也可以按需調(diào)用其中任意個方法:
mEdit.addTextChangedListener { onTextChanged { text, start, before, count -> println("onTextChanged") } }
不過此處的addTextChangedListener是一個擴展函數(shù),需要我們來自己實現(xiàn):
inline fun TextView.addTextChangedListener(init: TextWatcherBridge.() -> Unit) = addTextChangedListener(TextWatcherBridge().apply(init)) class TextWatcherBridge : TextWatcher { private var beforeTextChanged: ((CharSequence?, Int, Int, Int) -> Unit)? = null private var onTextChanged: ((CharSequence?, Int, Int, Int) -> Unit)? = null private var afterTextChanged: ((Editable?) -> Unit)? = null override fun beforeTextChanged(s: CharSequence?, start: Int, count: Int, after: Int) { beforeTextChanged?.invoke(s, start, count, after) } override fun onTextChanged(s: CharSequence?, start: Int, before: Int, count: Int) { onTextChanged?.invoke(s, start, before, count) } override fun afterTextChanged(s: Editable?) { afterTextChanged?.invoke(s) } fun beforeTextChanged(listener: (CharSequence?, Int, Int, Int) -> Unit) { beforeTextChanged = listener } fun onTextChanged(listener: (CharSequence?, Int, Int, Int) -> Unit) { onTextChanged = listener } fun afterTextChanged(listener: (Editable?) -> Unit) { afterTextChanged = listener } }
原理就是實現(xiàn)一個擴展函數(shù),把我們自己實現(xiàn)的TextWatcherBridge加入到回調(diào)中,因為Kotlin支持函數(shù)式編程,里面都是高階函數(shù)。為了減少性能損耗,擴展函數(shù)聲明為內(nèi)聯(lián)函數(shù)。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
Android中獲取sha1證書指紋數(shù)據(jù)的方法
大家都知道在Android開發(fā)中,經(jīng)常要獲取sha1證書指紋,所以這篇文章主要介紹在Android中如何使用命令獲取sha1證書指紋數(shù)據(jù)的方法,有需要的可以參考借鑒。2016-09-09Flutter應(yīng)用程序?qū)崿F(xiàn)隱私屏幕示例解析
這篇文章主要為大家介紹了Flutter應(yīng)用程序?qū)崿F(xiàn)隱私屏幕示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-09-09Android Vitamio和ExoPlayer兩種播放器優(yōu)劣分析
Vitamio和ExoPlayer都是用于安卓平臺的視頻播放器庫,它們各有優(yōu)缺點,具體使用哪一個,需要根據(jù)你的實際需求、開發(fā)經(jīng)驗、項目規(guī)模等多個因素綜合考慮2023-04-04解決Android SELinux權(quán)限問題記錄分析
這篇文章主要為大家介紹了解決Android SELinux權(quán)限問題記錄分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-11-11Android中backgroundDimEnabled的作用
這篇文章主要介紹了Android中backgroundDimEnabled的作用的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解掌握這部分內(nèi)容,需要的朋友可以參考下2017-10-10Flutter打包apk報錯Your?app?isn't?using?AndroidX解決
這篇文章主要為大家介紹了Flutter打包apk報錯Your?app?isn't?using?AndroidX解決方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-08-08Android高版本API方法如何在低版本系統(tǒng)上做兼容性處理淺析
這篇文章主要給大家介紹了關(guān)于Android高版本API方法如何在低版本系統(tǒng)上做兼容性處理的相關(guān)資料,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考借鑒,下面隨著小編來一起學習學習吧2018-08-08