建了一個面試題解答的項(xiàng)目,大家可以看一看,希望大家?guī)兔o一個star,謝謝了! 項(xiàng)目地址:https://github.com/NotFound9/interviewGuide?

起因
最近在復(fù)習(xí)iOS中的消息轉(zhuǎn)發(fā)機(jī)制,如果需要在動態(tài)方法解析這一階段對消息進(jìn)行處理,一般需要調(diào)用class_addMethod方法給類動態(tài)地增加方法,例如:

我當(dāng)時敲代碼的時候發(fā)現(xiàn)自己對于class_addMethod這個方法的第四個參數(shù)const char *types不太清楚,后來搜了一下類型編碼,發(fā)現(xiàn)網(wǎng)上的文章主要是對于屬性的類型編碼進(jìn)行了介紹,也沒有人對函數(shù)的類型編碼進(jìn)行介紹,就去看了一下文檔了解了一下。再后來發(fā)現(xiàn)介紹消息轉(zhuǎn)發(fā)機(jī)制的那篇文章中對于這個參數(shù)的傳入是錯誤的,

圖中添加的方法是void functionForMethod1(idself, SEL _cmd),
傳入的函數(shù)類型編碼是"@:",
而正確的函數(shù)類型編碼應(yīng)該是"v@:",
探究
下面讓我們來看看這個方法:
BOOL class_addMethod(Class cls, SEL name, IMP imp, const char *types);? ? ??
這個方法主要接受四個參數(shù)
Class cls 要添加方法的類
SEL name 被添加方法的名字
IMP imp 添加的方法的實(shí)現(xiàn)
const char *types?描述方法參數(shù)類型的字符數(shù)組。?
描述方法參數(shù)類型的字符數(shù)組的第一個字符是代表返回值的類型,后面的字符依次代表參數(shù)的類型,因?yàn)镺bjective-C中的函數(shù)會包含兩個隱式參數(shù),也就是方法調(diào)用者和方法名,例如
+(void)method
實(shí)際應(yīng)該是
void?method(id self, SEL _cmd)
如果返回值為空,那么函數(shù)的類型編碼的第一個字符是v,如果不為空,則為返回值類型對應(yīng)的編碼,詳細(xì)的可以看下面的編碼對應(yīng)表
因?yàn)榈谝粋€參數(shù)是方法調(diào)用者,它的類型肯定是對象類型,所以類型編碼的第二個字符一定是@
因?yàn)榈诙€參數(shù)是方法名的類型,第三個字符一定是:
所以這個函數(shù)
void?method(id self, SEL _cmd)????????
的類型編碼為 "v@:"
那么如果要添加的函數(shù)是一個set函數(shù),類型編碼是怎么樣的呢?
-(void)setA:(NSString *)a
同理,set方法實(shí)際的函數(shù)是這樣的:
void?setA(id self, SEL _cmd, id a)??
與上面無參數(shù)的方法相比,只是多了一個參數(shù),
所以類型編碼為"v@:@",代碼為
class_addMethod(self, @selector(setA:), (IMP)setA,"v@:@");
悔悟。。。
最后發(fā)現(xiàn)自己還是太年輕了。。。
蘋果還是考慮到了開發(fā)者手動寫函數(shù)的類型編碼容易出錯的情況,所以還是提供了api來獲取函數(shù)的類型編碼(其實(shí)我想說。。。既然能提供api獲取函數(shù)的類型編碼,為什么不在class_addMethod的實(shí)現(xiàn)中去獲取函數(shù)的類型編碼,而需要外部傳入)
主要是有這么一個函數(shù)
const char * _Nullablemethod_getTypeEncoding(Method _Nonnull m)?
通過傳入類名和,獲函數(shù)名,獲取函數(shù)的類型編碼,例如這樣:

類型編碼對應(yīng)表:
