Kotlin淺析延遲初始化與密封類的實現(xiàn)方法
一、lateinit延遲初始化關(guān)鍵字
Kotlin中很多語法特性,如變量不可變,變量不可為空,等等 這些特性都是為了盡可能地保證程序安全而設(shè)計的,比如你的類中存在很多全局變量實例,為了保證它們的能夠滿足Kotlin的空指針檢查語句標準,你不得不做非空判斷保護,即使你非常確定它們不會為空。
下面距離看一下 :
class MainActivity : AppCompatActivity() { private var s: String? = null override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) s = "test" Log.d("TAG", "onCreate: ${s!!.length}") } }
我們將s 設(shè)置為了全局變量 , 但是它的賦值工作在onCreate()方法進行的,因此不得不將s賦值為null。
雖然你確定在打印前已經(jīng)將s賦值成功,但是打印s的長度仍然要進行判空處理才行,否則編譯不通過。
當(dāng)你的代碼中越來越多的全局變量實例時,這個問題就會變得越來越明顯,到時候可能必須寫大量額外判空處理的代碼,卻只是為了滿足Kotlin的編譯要求。
幸運的是,這個問題有解決辦法的且非常之簡單,就是對全局變量進行延遲初始化。
初始化使用的關(guān)鍵字 lateinit ,它可以高速編譯器,我會在晚些時候?qū)@個變量進行初始化,這樣就不用一開始就賦值為null了。
class MainActivity : AppCompatActivity() { private lateinit var s: String override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) s = "test" Log.d("TAG", "onCreate: ${s.length}") } }
可以看到,加上了lateinit關(guān)鍵字 ,這樣一開始就不用賦值為null了,打印長度的時候也不用進行判空處理了,當(dāng)然使用lateinit關(guān)鍵字 也不是沒有風(fēng)險,如果沒有進行賦值,那么程序一定會崩潰,并拋出異常。
另外,我們可以通過代碼來判斷全局變量是否已完成了初始化工作,這樣某些時候可以有效避免重復(fù)對某一個變量進行初始化操作。
class MainActivity : AppCompatActivity() { private lateinit var s: String override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) if (!::s.isInitialized) { s = "test" } Log.d("TAG", "onCreate: ${s.length}") } }
二、使用密封類優(yōu)化代碼
新建一個Kotlin文件,代碼如下 :
interface Result class Success(val msg: String) : Result class Failure(val error: Exception) : Result
這里定義了一個Result接口,用于表示某個操作的執(zhí)行結(jié)果,接口中沒有編寫任何內(nèi)容。然后定義了兩個類去實現(xiàn)Result接口:一個Success類用于表示成功時結(jié)果,一個Failure類用于表示失敗時結(jié)果。
接下來再定義一個getResultMsg()方法 ,用于獲取最后執(zhí)行結(jié)果的信息,代碼如下:
fun getResultMsg(result: Result) = when (result) { is Success -> result.msg is Failure -> result.error.message else -> throw IllegalArgumentException() }
getResultMsg()方法中接受一個Result參數(shù),我們通過when語句來判斷,如果Result屬于Success就返回成功的消息,如果Result是Failure就返回錯誤的信息,到目前為止代碼是沒什么問題的,煩人的是不得不寫一個else條件語句,否則Kotlin編譯不通過,其實代碼永遠也走不到else里 因為只有兩種類型的存在,只是為了滿足Kotlin的語法而已。
另外,編寫else條件還有一個潛在的風(fēng)險,如果我們新增一個Unknown類并實現(xiàn)了Result接口,用于表示未知的執(zhí)行結(jié)果,但是忘記了在getResultMsg()方法中添加相應(yīng)的條件,編譯器不會提醒我們的,而是直接進入else條件里面去,從里面拋出異常并導(dǎo)致程序崩潰。
Kotlin的密封類很好的解決了此問題,密封類的關(guān)鍵字是sealed class ,它的用法同樣很簡單,我們可以輕松的將Result接口改造成密封類寫法:
sealed class Result class Success(val msg: String) : Result() class Failure(val error: Exception) : Result() fun getResultMsg(result: Result) = when (result) { is Success -> result.msg is Failure -> result.error.message }
代碼沒什么變化,只是將interface改成了 sealed class。密封類是一個可繼承的類,因此在繼承它的時候還需要加上一對括號。
密封類的優(yōu)點是,可以再getResultMsg() 方法中取消else條件語句。
為什么取消掉else條件語句還能編譯通過呢,Kotlin編譯器會自動檢查該密封類有哪些子類,并強制要求將每一個子類所對應(yīng)的條件全部處理。這樣就可以保證,即使沒有else條件,也不可能出現(xiàn)漏寫的情況,如果現(xiàn)在新增一個Unknown類,并也讓它繼承自Result,此時getResultMsg()方法就一定會報錯,必須添加Unknown語句條件才能編譯通過。
注意:密封類及其所有子類只能定義同一個文件的頂層位置,不能嵌套在其他類中,這也是被密封類底層實現(xiàn)機制所限制的。
到此這篇關(guān)于Kotlin淺析延遲初始化與密封類的實現(xiàn)方法的文章就介紹到這了,更多相關(guān)Kotlin延遲初始化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android Studio連接MySql實現(xiàn)登錄注冊(附源代碼)
登錄注冊是常用的一個功能,正好今天用android studio 做一個類似于這樣的登錄軟件,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05Android百度地圖定位后獲取周邊位置的實現(xiàn)代碼
這篇文章主要為大家詳細介紹了Android百度地圖定位后獲取周邊位置的實現(xiàn)代碼,準確獲取周邊地理位置,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-01-01Android上實現(xiàn)easyconfig(airkiss)方法
本篇文章主要給大家講解了在Android系統(tǒng)上實現(xiàn)easyconfig(airkiss)的方法,有這方面需要的朋友參考學(xué)習(xí)下吧。2018-01-01android ListView結(jié)合xutils3仿微信實現(xiàn)下拉加載更多
本篇文章主要介紹了android ListView結(jié)合xutils3仿微信實現(xiàn)下拉加載更多,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-11-11