欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Kotlin淺析延遲初始化與密封類的實現(xiàn)方法

 更新時間:2022年08月02日 14:43:59   作者:m0_63970488  
Kotlin語言的許多特性,包括變量不可變,變量不可為空,等等。這些特性都是為了盡可能地保證程序安全而設(shè)計的,但是有些時候這些特性也會在編碼時給我們帶來不少的麻煩,下面我們來了解延遲初始化和密封類的特點

一、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)文章

最新評論