????人總是要進步的~最近在學(xué)kotlin,號稱是新貴,不明覺厲乍迄,慢慢學(xué)習(xí)吧。
kotlin 的簡介里都會有一條是空安全, 尤其是用來對比java士败,今天特意看了下,寫了篇墨跡的渣渣褥伴,算是總結(jié)吧谅将。
????ok,首先說 空 這件事重慢,空 :One of the most common pitfalls in many programming languages, including Java is that of accessing a member of a nullreferences, resulting in null reference exceptions. In Java this would be the equivalent of a NullPointerException or NPE forshort. from kotlin-doc饥臂,簡而言之:對空對象的引用會觸發(fā)NEP。那么就有兩種方式去解決問題似踱,一種是在每次引用對象的時候都對其進行空安全的檢測隅熙,另外一種就是通過手段(代碼檢測稽煤,編譯檢查 各種黑科技)來強制保證對象不為空。
???? 比較一下這兩種方法囚戚,第一種似乎就是java酵熙,對 對象是否null 要求不嚴格,需要有工程師(程序猿)自行斟酌驰坊,也可以對每個對象都加 if 判斷匾二。另外一種看起來好一些,可以保證放心的引用拳芙,但是會存在問題察藐,比如我需要一個全局變量,但是我不知道后續(xù)的操作是否會對其進行賦值舟扎,萬一對其賦值了分飞,那么我需要展示它。僅就此而言睹限,那么聲明的所有對象都不為空這件事可能是做不到的了譬猫。
????看了kotlin發(fā)現(xiàn)它將這兩種情況給糅合了下,它允許聲明一定不為空的對象邦泄,也允許可能為空的對象删窒。通過對兩種對象的不同處理操作來保證其是空安全的。kotlin 區(qū)分了這兩類對象顺囊,對于聲明為一定不為空的變量肌索,kotlin執(zhí)行嚴格的代碼檢測來保證其在任何時候都不會被賦予空值,這就實現(xiàn)上面說的第二種思路特碳。如果你確定以及肯定你的對象一定是不為空的話诚亚,聲明成非空類型變量吧,kotlin也會幫你監(jiān)視它的午乓。對于可能為空的對象站宗,kotlin也支持你將其聲明為允許空值得對象,但是在你調(diào)用它的時候益愈,以string為例梢灭,即便你在上一行寫了這么一句var str: String? = "Kotlin baba"
,下一行print(str.get(0))
,也是不能通過代碼檢測的蒸其,飄紅敏释,報錯,會強制你去做判空 操作摸袁。而正是通過這這兩種方式钥顽,kotlin實現(xiàn)了空安全。
????首先所有的變量在聲明的時候都會決定一件事 這個變量是否可以(可能)是空值靠汁,這會影響到后續(xù)對這個變量引用時的一系列判斷蜂大,其實就是你聲明了一個變量闽铐,如果你既沒有說明他是可空的,也沒有在引用的時候做安全處理(后面會談到具體的安全處理)奶浦,ok兄墅,編譯不會通過,強制保證 空安全财喳。身為程序猿察迟,還是要拿代碼說的,下面給出一下java 和 kotlin 的比較,主要是對可能造 NEP 的對比(注:相同的代碼耳高,上邊的是java扎瓶,下邊的是kotlin。)泌枪。首先說變量的空安全概荷,In java,聲明一個變量并對其進行重新賦值使用 like this:
//java code
private static void printNEP() {
String str = "I will Be Null";
/**
* .... 對 str 各種操作碌燕,拼接误证,轉(zhuǎn)換,網(wǎng)絡(luò)請求重賦值 巴拉巴拉
*/
str = null;
System.out.println(str.charAt(1024));
//charAt 會被lint 標黃出來可能會造成NEP修壕,但是愈捅,并不會強制你改
}
see kotlin
為了看到紅色,不能用代碼了撒慈鸠,翻譯如下 空值不能被賦給非空的類型 string蓝谨,先不解釋,可以看到下面一行也是紅的青团,
32,33兩行都紅了譬巫,所以首先肯定是不能編譯了,所以這就涉及到kotlin空安全了督笆,先說32行的飄紅芦昔,開頭說過,kotlin在聲明變量的時候 會強制要求你說明 這個變量是否可以(可能)是空值娃肿,
var str: String = "I will Be Null";
咕缎,這種聲明方式就是這個string變量一定不為空的意思,一旦存在可能會給這個變量賦空值的操作料扰,就會飄紅了锨阿。ok那么你可能就是想用一個可以空的值,你可以這樣聲明
一個?就表示這個變量可以為空记罚,32行的飄紅消失,對一個可以為空的變量賦空值時OK的壳嚎,沒毛病吧桐智,但是33行的飄紅依然存在末早,我就不截圖了,異常提示信息跟上邊(圖2)的一樣说庭,再來翻譯一波然磷,翻譯如下only 安全 或者 非空斷言是被允許的在一個接受string的接受者,官方翻譯并不知道~但是個人理解就是你想使用一個有可能是空值的變量前必須對它進行判空操作
1.條件判空 : 這個就像我們經(jīng)常做的 if(null != xxx)
刊驴,然后咋樣咋樣
2.安全調(diào)用:這個就是圖二里邊給出的第一個提示?.姿搜,這個東西呢,類似 于java的?:
捆憎,舅柜?后邊是不為空的情況下的正確的返回值,:后邊就是返回空躲惰,
這個東西打印null致份,而不是NEP,如果str的1024不為空础拨,get返回它氮块,如果為空,返回null诡宗,而不是引用空值導(dǎo)致NEP滔蝉。
3.Elvis操作符(*Groovy里有一種操作符叫Elvis運算符,可以簡化Java里的二元操作符塔沃。如果我們需要在某個值是空或false時指定一個默認值時蝠引,使用Elvis運算符會使表達式更簡潔。這里的false判斷遵循Groovy的Truth判定方法:把null芳悲,空字符串立肘,空集合視為false 。from http://zjumty.iteye.com/blog/1882380 *):就是名扛?:的簡化版谅年,print(str?.get(1024)?:"null or what ever u want")
4.!!.,你成功的得到了一個崩潰
這個操作符會給你一個非空值讓你繼續(xù)引用肮韧,或者干脆就拋出NEP崩潰啦啦融蹂。
總結(jié)一下
為什么說kotlin是空安全的呢,個人理解是因為它能明確的告訴你這個值是不是可能為空弄企,如果不為空超燃,那么它一定不是空的,而且過程中也不許對其賦予空值拘领,這樣無論何時你使用它的時候都不會引發(fā)NEP意乓。而不像java那樣,即使你在聲明的時候?qū)ζ溥M行賦值约素,但是在各種操作中還是可能將其賦值為空届良。然后在調(diào)用的時候NEP笆凌。如果在kotlin中聲明了一個可以為空的值,那就注定了你再使用這個值得時候必須要對其進行判空 操作士葫,進而避免NEP乞而,kotlin 靜態(tài)的代碼檢測就保證了其是空安全的,java的檢測只是提醒你慢显,即便你明確的知道這個東西一定不會為空爪模。
ps
https://www.gitbook.com/book/huanglizhuo/kotlin-in-chinese/details
奉上鏈接