在 Objective-C 中向 nil 發(fā)送消息是完全有效的——只是在運行時不會有任何作用:
如果一個方法返回值是一個對象,那么發(fā)送給nil的消息將返回0(nil)。例如:
Person* motherInlaw =[[aPerson spouse]mother];
如果 spouse 對象為 nil,那么發(fā)送給 nil 的消息 mother 也將返回 nil。
如果方法返回值為指針類型,其指針大小為小于或者等于sizeof(void*),float,double,long double 或者 long long 的整型標量,發(fā)送給 nil 的消息將返回0。
如果方法返回值為結(jié)構(gòu)體,發(fā)送給 nil 的消息將返回0。結(jié)構(gòu)體中各個字段的值將都是0。
如果方法的返回值不是上述提到的幾種情況,那么發(fā)送給 nil 的消息的返回值將是未定義的。
具體原因如下:
objc是動態(tài)語言,每個方法在運行時會被動態(tài)轉(zhuǎn)為消息發(fā)送,即:objc_msgSend(receiver, selector)。
那么,為了方便理解這個內(nèi)容,還是貼一個objc的源代碼:
// runtime.h(類在runtime中的定義)// http://weibo.com/luohanchenyilong/// https://github.com/ChenYilongstructobjc_class{Class isa OBJC_ISA_AVAILABILITY;//isa指針指向Meta Class,因為Objc的類的本身也是一個Object,為了處理這個關(guān)系,runtime就創(chuàng)造了Meta Class,當給類發(fā)送[NSObject alloc]這樣消息時,實際上是把這個消息發(fā)給了Class Object#if!__OBJC2__Class super_class OBJC2_UNAVAILABLE;// 父類constchar*name OBJC2_UNAVAILABLE;// 類名longversion OBJC2_UNAVAILABLE;// 類的版本信息,默認為0longinfo OBJC2_UNAVAILABLE;// 類信息,供運行期使用的一些位標識longinstance_size OBJC2_UNAVAILABLE;// 該類的實例變量大小structobjc_ivar_list*ivarsOBJC2_UNAVAILABLE;// 該類的成員變量鏈表structobjc_method_list**methodListsOBJC2_UNAVAILABLE;// 方法定義的鏈表structobjc_cache*cacheOBJC2_UNAVAILABLE;// 方法緩存,對象接到一個消息會根據(jù)isa指針查找消息對象,這時會在method Lists中遍歷,如果cache了,常用的方法調(diào)用時就能夠提高調(diào)用的效率。structobjc_protocol_list*protocolsOBJC2_UNAVAILABLE;// 協(xié)議鏈表#endif} OBJC2_UNAVAILABLE;
objc在向一個對象發(fā)送消息時,runtime庫會根據(jù)對象的isa指針找到該對象實際所屬的類,然后在該類中的方法列表以及其父類方法列表中尋找方法運行,然后在發(fā)送消息的時候,objc_msgSend方法不會返回值,所謂的返回內(nèi)容都是具體調(diào)用時執(zhí)行的。那么,回到本題,如果向一個nil對象發(fā)送消息,首先在尋找對象的isa指針時就是0地址返回了,所以不會出現(xiàn)任何錯誤。
作者:啟發(fā)禪悟
鏈接:http://www.itdecent.cn/p/b7cda433e4f5
來源:簡書
著作權(quán)歸作者所有。商業(yè)轉(zhuǎn)載請聯(lián)系作者獲得授權(quán),非商業(yè)轉(zhuǎn)載請注明出處。