iOS重名category 的調用方式

今天看了一篇文章【iOS】category 重寫方法的調用,介紹了category在重寫主類方法,和多個category中方法重名的時候的調用方式。其中提到category加入函數列表中的順序是反向于文件編譯的順序,即編譯是根據buildPhases->Compile Sources里面的順序從上至下編譯的,那么category的執行順序就是反向于這個順序的。那么我們來驗證一下。

@interface Father : NSObject

- (void)name;

@end

@implementation Father

- (void)name{
    NSLog(@"my name is father");
}
@end
@interface Father (aaa)

@end

@implementation Father (aaa)

- (void)name{
    NSLog(@"my name is father aaa");
}

@end
@interface Father (bbb)

@end

@implementation Father (bbb)

- (void)name{
    NSLog(@"my name is father bbb");
}
@end
    int count;
    Method* list;
    list = class_copyMethodList( [Father class], &count );
    for (int i = 0; i < count; i++) {
        NSLog(@"%@",NSStringFromSelector(method_getName(list[i])));
    }

運行結果:

2018-01-04 23:59:59.904820+0800 test[48010:12449543] name:
2018-01-04 23:59:59.904937+0800 test[48010:12449543] name:
2018-01-04 23:59:59.905057+0800 test[48010:12449543] name:

打印出3個name:,可以看到category和主類的方法都在函數列表中,但是selector的名字都一樣啊,那么怎么確定他的順序呢?哈哈,我們來想想辦法,看看method都包括哪些內容

struct objc_method {
    SEL _Nonnull method_name                                 OBJC2_UNAVAILABLE;
    char * _Nullable method_types                            OBJC2_UNAVAILABLE;
    IMP _Nonnull method_imp                                  OBJC2_UNAVAILABLE;
}

selector是查找函數的key,而selector本身就是一個字符串,那么我們可以定義同樣的selector,通過定義不同的method_types加以區分。

@interface Father : NSObject
- (void)name:(int)a;
@end

@implementation Father

- (void)name:(int)a{
    NSLog(@"my name is father");
}
@end
@interface Father (aaa)

@end

@implementation Father (aaa)

- (void)name:(char)a{
    NSLog(@"my name is father aaa");
}

@end
@interface Father (bbb)

@end

@implementation Father (bbb)

- (void)name:(short)b{
    NSLog(@"my name is father bbb");
}
@end
    int a;
    Father* father = [[Father alloc]init];
    [father name:a];

    int count;
    Method* list;
    list = class_copyMethodList( [Father class], &count );
    des = method_getDescription(list);
    for (int i = 0; i < count; i++) {
        NSLog(@"%@ %s",NSStringFromSelector(method_getName(list[i])), method_getTypeEncoding(list[i]));
    }

運行結果:

2018-01-05 00:27:44.160124+0800 test[48427:12478941] my name is father bbb
2018-01-05 00:21:44.170938+0800 test[48427:12478941] name: v20@0:8s16
2018-01-05 00:21:44.171095+0800 test[48427:12478941] name: v20@0:8c16
2018-01-05 00:21:44.200842+0800 test[48427:12478941] name: v20@0:8i16

其中8s中的s代表short,8c中的c代表char,8i中的i是int。

到Compile Sources中看一下,發現編譯的順序是

Father+aaa.m

Father+bbb.m

Father.m

說明是先編譯的Father.m,然后編譯的Father+aaa.m和Father+bbb.m,所以Father+bbb.m中的函數

-(void)name:(short)b 最后一個載入,也得到了最終執行。

接下來我們在Compile Sources中拖動Father.m交換文件順序。

Father+bbb.m

Father.m

Father+aaa.m

運行結果:

2018-01-05 00:37:28.714927+0800 test[48698:12497028] my name is father aaa
2018-01-05 00:37:28.715230+0800 test[48698:12497028] name: v20@0:8c16
2018-01-05 00:37:28.727622+0800 test[48698:12497028] name: v20@0:8s16
2018-01-05 00:37:28.727735+0800 test[48698:12497028] name: v20@0:8i16

可以看到,Father+aaa.m和Father+bbb.m中的函數在methodlist中的順序交換了,最終執行的也成了Father+aaa.m中的函數。但是主類的函數(name: v20@0:8i16)一直是最先放入methodlist中的。這也說明了category的加載是在主類之后,這與Compile Sources中的順序無關。

再思考2個問題:

  1. 子類和父類都實現了相同的category函數,會執行誰呢?
  2. 子類繼承后重寫了父類的category函數,會執行誰呢?

其實答案很明顯,按照selector的查找順序,會先在子類中找,再到父類中找。只要子類找到了selector,不管是不是category,都會先執行。

看下這兩個問題的代碼:

@implementation Father (bbb)

- (void)name:(double)b{
    NSLog(@"my name is father bbb");
}

- (void)age{
    NSLog(@"I am 30 years old");
}
@end
@interface Son : Father
- (void)age;
@end

@implementation Son

- (void)name:(int)a{
    NSLog(@"my name is son");
}

- (void)age{
    NSLog(@"I am 6 years old");
}
@end
@interface Son (bbb)

@end

@implementation Son (bbb)
- (void)name:(double)a{
    NSLog(@"my name is son bbb");
}
@end
Son* son = [[Son alloc]init];
[son name:a];

執行結果:

2018-01-05 10:29:15.625968+0800 test[50457:12615463] my name is son bbb
2018-01-05 10:29:15.626104+0800 test[50457:12615463] I am 6 years old

明顯,子類son的category得到了執行,復寫的age函數也是先執行的子類的,與分析的結果一致。

category導致的問題

從上面的結果中我們可以看到,category雖然給我們提供了便利,但是最大的問題就是不確定性,當出現重名的函數時,執行的結果很可能和預想的不一樣。尤其在一些大的工程中,代碼多,很可能出現重名的category。特別是很多工程采用sdk組件化集成,內部的代碼都不知道實現了哪些category,當出現一些crash和執行錯誤的時候,很可能是執行了不同的category導致的。那么怎么控制呢?

  1. 首先是命名,category中要有自己的命名規范,根據category的名字給函數加前綴。由于objective-c沒有namespace,只能通過前綴來區分。
@implementation Father (aaa)

- (void)aaa_Name:(char)a{
    NSLog(@"my name is father aaa");
}

@end

@implementation Father (bbb)

- (void)bbb_Name:(char)b{
    NSLog(@"my name is father bbb");
}

@end
  1. 對于一些公用的比較基礎的category,放到基礎庫中,大家引用同一份,不要各自定義。
@interface NSString (Addition)
- (NSString *)urlEncode;
- (NSString *)md5Digest;
@end
  1. category的使用時機。個人認為,有2點:
    • 如果要實現的功能,是對這個類普遍生效的,則使用category。如果是對單獨場景的一種擴展,還是使用繼承比較好。其實這種討論類似于一個方法是要加到基類中,還是繼承后實現到子類中。比如UIView,添加動畫,frame設置這些基礎方法,需要放到category中。像UIButton這中針對單獨場景的特殊設置,就用的子類。
    • 在大工程多個sdk組件化的工程中,對于其他模塊封裝的類,還是盡量使用子類化,category還是盡量實現在聲明這個類的sdk中。
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,533評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,055評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,365評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,561評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,346評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,889評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,978評論 3 439
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,118評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,637評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,558評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,739評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,246評論 5 355
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 43,980評論 3 346
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,362評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,619評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,347評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,702評論 2 370

推薦閱讀更多精彩內容

  • 轉至元數據結尾創建: 董瀟偉,最新修改于: 十二月 23, 2016 轉至元數據起始第一章:isa和Class一....
    40c0490e5268閱讀 1,745評論 0 9
  • 他:“遇見問題了,請幫我分析一下?” 我:“好的,請說。” 他:這個大客戶連續拜訪了兩次(副總),并贈送小禮品,在...
    瘋狂的小蝸牛閱讀 690評論 0 0
  • 結束完一周的工作,周末終于可以約上小伙伴去探店了,今天要給大家介紹的是這家小而精的吉川堂。說真的,如果真的可以當一...
    喜樂喜樂love閱讀 413評論 0 0
  • 從最初到現在,你,換了多少個手機?或許,你一時答不上來,或許,你思考半天還是一無所獲。但是,多數人還是會記得自己擁...
    桐妍很無忌閱讀 327評論 0 0
  • 文/極客少年 奧古末紀,群王并起,龍武覺醒,諸圣爭霸;東方軍團喋血長空,冬國之海風起云涌。遠古神殿流傳著一則不朽的...
    極客少年閱讀 557評論 0 4