在iOS開發過程中經常需要與服務器進行數據通訊,Json就是一種常用的高效簡潔的數據格式。
JSON建構有兩種結構:
json簡單說就是javascript中的對象和數組,所以這兩種結構就是對象和數組2種結構,通過這兩種結構可以表示各種復雜的結構
1、對象:對象在js中表示為“{}”擴起來的內容,數據結構為 {key:value,key:value,...}的鍵值對的結構,在面向對象的語言中,key為對象的屬性,value為對應的屬性值,所以很容易理解,取值方法為對象.key 獲取屬性值,這個屬性值的類型可以是 數字、字符串、數組、對象幾種。
2、數組:數組在js中是中括號“[]”擴起來的內容,數據結構為 ["java","javascript","vb",...],取值方式和所有語言中一樣,使用索引獲取,字段值的類型可以是數字、字符串、數組、對象幾種。
經過對象、數組2種結構就可以組合成復雜的數據結構了。
問題現象
但是幾個項目下來一直遇到一個坑爹的問題,程序在獲取某些數據之後莫名崩潰。其實很早就發現了原因:由於服務器的數據庫中有些字段為空,然後以Json形式返回給客戶端時就會出現這樣的數據:
"somevalue":null
通過JsonKit 這個第三方庫解析出來的數據就成了
somevalue = "<null>";
這個數據類型不是nil 也不是 String。 解析成對象之後,如果直接向這個對象發送消息(eg:length,count 等等)就會直接崩潰。提示錯誤為:
-[NSNull length]: unrecognized selector sent to instance 0x388a4a70
解決方法
其實一直沒有找到完美的解決辦法,坑了我很久。
1、最開始的解決方法就是為了應付當前遇到的崩潰,看看哪個字段可能為空,那麼就對該字段使用前進行判斷,通過崩潰時的錯誤提示可以看出,這樣的字段解析成的對象是 NSNull 類型的,所以可以直接判斷是不是此類型:
if (![isKindOfClass:[NSNull class]]){xxxxxxx;}
因為字段實在太多,就找一個補一個。
2、後來想徹底解決這問題,就打算從數據源下手,其實應該可以用正則表達式匹配這個null ,然後進行替換,奈何正則是我的硬傷啊。於是就相出了一個山寨方法:字符串匹配。在獲取到服務器返回的Json時,返回結果時string對象,於是就先替換 null 為 為空字符””,然後再解析。
json = [jsonStr stringByReplacingOccurrencesOfString:@":null" withString:@":\"\""];
這個方法本來很奏效,但是我這裡的服務器返回極不簡潔,各種垃圾數據(不吐槽這了)。。。反正這樣會導致json無法解析了。
3、最後沒有辦法,只能在解析的時候下手,把是NSNull 類型的值替換成nil。 一般就寫個tool方法,然後解析時調用。但是嫌太麻煩,就想弄寫個宏,通過搜索驚奇的發現宏也是可以有返回值的,結果如下:
#define VerifyValue(value)\
({id tmp;\
if ([value isKindOfClass:[NSNull class]])\
tmp = nil;\
else\
tmp = value;\
tmp;\
})\
宏裡的最後一句語句便是返回值。然後在解析數據時調用宏:
contact.contactPhone = VerifyValue(contactDic[@"send_ContactPhone"]);
4、如果你使用AFNetwork 這個庫做網絡請求的話,可以用以下代碼,自動幫你去掉這個討厭的空值
self.removesKeysWithNullValues = YES;
5、終極方案
終於找到了一勞永逸的方案,牛逼的老外寫了一個Category,叫做NullSafe ,在運行時操作,把這個討厭的空值置為nil,而nil是安全的,可以向nil對象發送任何message而不會奔潰。這個category使用起來非常方便,只要加入到了工程中就可以了,你其他的什麼都不用做,對,就是這麼簡單。詳細的請去Github上查看;