kotlin代碼中!!操作符的清楚方法

空安全是Kotlin中非常實(shí)用的特性,它能夠讓你避免許多隱藏的NullPointerExceptions問題。然而當(dāng)你使用AS將Java代碼轉(zhuǎn)換成Kotlin代碼后會(huì)出現(xiàn)很多的!! 操作符,因?yàn)槌霈F(xiàn)!!意味著這里可能有未捕獲的KotlinNullPointerException異常。

對(duì)于一個(gè)有強(qiáng)迫癥的程序猿來說一個(gè)類中大篇幅的出現(xiàn)!! 絕對(duì)會(huì)使他們發(fā)瘋的。幸運(yùn)的是我們可以通過以下方式避免使用!!操作符。文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

1. 使用val代替var文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

我們都知道val修飾的變量是只讀的,var修飾的變量是可變的,所以我們應(yīng)該盡可能多的使用val,因?yàn)槭褂胿al修飾的變量必須有一個(gè)value,這樣你就不用再擔(dān)心空指針的問題,所以如果你能確定一個(gè)變量可以定義成一個(gè)常量,那么請(qǐng)用val修飾它。文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

2. 使用lateinit文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

然而你并不能把所有的變量用val修飾,,比如那些需要在Activity的onCreate()中進(jìn)行初始換的變量,針對(duì)這個(gè)情況你可以考慮使用lateinit修飾變量,例如下面的代碼文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

1
2
3
4
5
6
7
8
9
private var mAdapter: RecyclerAdapter<Transaction>? = null
override fun onCreate(savedInstanceState: Bundle?) {
?super.onCreate(savedInstanceState)
?mAdapter = RecyclerAdapter(R.layout.item_transaction)
}
fun updateTransactions() {
?mAdapter!!.notifyDataSetChanged()
}

把mAdapter使用lateinit修飾后代碼變成這樣文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

1
2
3
4
5
6
7
8
9
10
private lateinit var mAdapter: RecyclerAdapter<Transaction>
override fun onCreate(savedInstanceState: Bundle?) {
?super.onCreate(savedInstanceState)
?mAdapter = RecyclerAdapter(R.layout.item_transaction)
}
fun updateTransactions() {
?mAdapter.notifyDataSetChanged()
}

需要注意的是如果訪問一個(gè)還沒有初始化的變量或?qū)傩詫?huì)導(dǎo)致UninitializedPropertyAccessException異常。文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

還有一點(diǎn)需要注意的是lateinit并不能修飾基本數(shù)據(jù)類型的變量或?qū)傩裕热鏘nt,Boolean等等,它會(huì)提示你文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

‘lateinit'modifier is nor allowed on properites of primitive types文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

此時(shí)你可以使用下面這個(gè)方式文章源自四五設(shè)計(jì)網(wǎng)-http://www.4968ejs.cn/40840.html

1
private var mNumber: Int by Delegates.notNull<Int>()

3. 使用let函數(shù)

下面這段代碼的提示我們經(jīng)常會(huì)見到

kotlin代碼中!!操作符的清楚方法-1

studio提醒我們mPhotoUrl的value在執(zhí)行uploadPhoto時(shí)可能已經(jīng)改變,不能確定是否非空,通常我們的解決方式是這樣的

1
2
3
4
5
6
7
private var mPhotoUrl: String? = null
fun uploadClicked() {
?if (mPhotoUrl != null) {
?uploadPhoto(mPhotoUrl!!)
?}
}

然而如果你不想使用!! 這里還有一種更優(yōu)雅的方式

1
2
3
4
5
private var mPhotoUrl: String? = null
fun uploadClicked() {
?mPhotoUrl?.let { uploadPhoto(it) }
}

只有當(dāng)mPhotoUrl不為空時(shí)let中的代碼才會(huì)執(zhí)行

如果你對(duì)let函數(shù)還不了解,可以看我寫的這篇文章

//www.jb51.net/article/131427.htm

4. 使用特定的函數(shù)處理復(fù)雜的場景

對(duì)于一些簡單的場景l(fā)et函數(shù)是很好用的,就像上面的情況,但是對(duì)于一些復(fù)雜的場景,比如下面的代碼

1
2
3
if (mUserName != null && mPhotoUrl != null) {
?uploadPhoto(mUserName!!, mPhotoUrl!!)
}

當(dāng)然你也可以使用let的方式處理,但是這樣代碼的可讀性就會(huì)降低了,這時(shí)候你可以定義一些特定的函數(shù)來解決這個(gè)問題。

比如下面這個(gè)可以判斷兩個(gè)參數(shù)非空的函數(shù)

1
2
3
4
5
fun <T1, T2> ifNotNull(value1: T1?, value2: T2?, bothNotNull: (T1, T2) -> (Unit)) {
?if (value1 != null && value2 != null) {
?bothNotNull(value1, value2)
?}
}

這樣你的代碼就變成了下面這樣

1
2
3
4
ifNotNull(mUserName, mPhotoUrl) {
?userName, photoUrl ->
?uploadPhoto(userName, photoUrl)
}

See,!!操作符消失了。

5. 使用Elvis操作符

對(duì)于那些必有返回的情況,Elvis非常的實(shí)用。

Elvis操作符,?:左邊的返回值不為空則返回,否則返回?:右邊的值

1
2
3
4
5
6
7
fun getUserName(): String {
?if (mUserName != null) {
?return mUserName!!
?} else {
?return "Anonymous"
?}
}

使用Elvis操作符后

1
2
3
fun getUserName(): String {
?return mUserName ?: "Anonymous"
}

通過上面這幾種辦法基本上你可以清除程序中所有的!!操作符了,而且你的代碼也會(huì)變得更加健壯。如果你還有其他的方式請(qǐng)?jiān)谠u(píng)論區(qū)留言吧。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流

繼續(xù)閱讀
我的微信
微信掃一掃
weinxin
我的微信
惠生活福利社
微信掃一掃
weinxin
我的公眾號(hào)
 

發(fā)表評(píng)論

匿名網(wǎng)友
:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

拖動(dòng)滑塊以完成驗(yàn)證