--存在類別關(guān)系時initialize和load的調(diào)用問題--
1,類別不會調(diào)用類的load方法
示例代碼一
@interface SuperClass : NSObjec
@end
@implementation SuperClass
+ (void) load {
NSLog(@"%@ %s", [self class], __FUNCTION__);
}
@end
@interface SuperClass (OneClass)
@end
@implementation ChildClass
+ (void) load {
NSLog(@"%@ %s", [self class], __FUNCTION__);
}
@end
@interface SuperClass (TowClass)
@end
@implementation SuperClass (TowClass)
+ (void) load {
NSLog(@"%@ %s", [self class], __FUNCTION__);
}
@end
示例代碼一中
OneClass和TowClass都是SuperClass的類別搜贤,并且都重寫了SuperClass的load方法短蜕。
現(xiàn)在我們在Xcode的項目中只簡單的import這幾個類鸡号,而不去使用它們。然后運行項目继榆,獲得下面的結(jié)果:
//示例代碼一
SuperClass +[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
SuperClass +[SuperClass(OneClass) load]
此時:如果我們刪除掉OneClass類別中重寫的load方法鸠信,重新運行項目舟山,獲得下面的結(jié)果:
SuperClass +[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
根據(jù)打印結(jié)果拢锹,可以看出:
如果OneClass沒有重寫load方法,則其不會執(zhí)行SuperClass的load方法兽叮。所以芬骄,類別中的load和繼承中的load是一樣的,只不過Category的load會在主類的load調(diào)用之后才會執(zhí)行鹦聪。
2,類別會覆蓋類的initialize方法
示例代碼二
@interface SuperClass : NSObject
@end
@implementation SuperClass
+(void)initialize{
NSLog(@"%@ %s",[self class], __FUNCTION__);
}
+ (void) load {
NSLog(@"%@ %s", [self class], __FUNCTION__);
}
@end
@interface SuperClass (OneClass)
@end
@implementation SuperClass (OneClass)
+(void)initialize{
NSLog(@"%@ %s",[self class], __FUNCTION__);
}
+ (void) load {
NSLog(@"%@ %s", [self class], __FUNCTION__);
}
@end
@interface SuperClass (TowClass)
@end
@implementation SuperClass (TowClass)
+(void)initialize{
NSLog(@"%@ %s",[self class], __FUNCTION__);
}
+ (void) load {
NSLog(@"%@ %s", [self class], __FUNCTION__);
}
@end
示例代碼二中:
兩個類別OneClass和TowClass都重寫了load方法和initialize方法账阻。這里重寫load是為了觸發(fā)initialize方法。
現(xiàn)在我們在Xcode的項目中只簡單的import這幾個類泽本,而不去使用它們宰僧。然后運行項目,獲得下面的結(jié)果:
//示例代碼二
SuperClass +[SuperClass(OneClass) initialize]
SuperClass +[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
SuperClass +[SuperClass(OneClass) load]
此時我們注釋掉OneClass重寫的initialize方法,再次運行程序琴儿,獲得下面的結(jié)果:
//示例代碼二
SuperClass +[SuperClass(TowClass) initialize]
SuperClass +[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
SuperClass +[SuperClass(OneClass) load]
接著我們注釋掉TowClass重寫的initialize方法,再次運行程序嘁捷,獲得下面的結(jié)果:
//示例代碼二
SuperClass +[SuperClass initialize]
SuperClass +[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
SuperClass +[SuperClass(OneClass) load]
最后我們注釋掉SuperClass的initialize方法造成,再次運行程序,獲得下面的結(jié)果:
//示例代碼二
+[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
SuperClass +[SuperClass(OneClass) load]
OK雄嚣,有沒有發(fā)現(xiàn)什么規(guī)律晒屎?先別著急說,接著往下看缓升。
現(xiàn)在鼓鲁,我們放開TowClass的注釋,再次運行程序港谊,獲得下面的結(jié)果:
//示例代碼二
SuperClass +[SuperClass(TowClass) initialize]
SuperClass +[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
SuperClass +[SuperClass(OneClass) load]
然后我們放開OneClass的注釋骇吭,再次運行程序,獲得下面的結(jié)果:
//示例代碼二
SuperClass +[SuperClass(OneClass) initialize]
SuperClass +[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
SuperClass +[SuperClass(OneClass) load]
最后我們放開SuperClass的注釋歧寺,再次運行程序燥狰,獲得下面的結(jié)果:
//示例代碼二
SuperClass +[SuperClass(OneClass) initialize]
SuperClass +[SuperClass load]
SuperClass +[SuperClass(TowClass) load]
SuperClass +[SuperClass(OneClass) load]
好了,現(xiàn)在你可以大膽的猜測斜筐,在分類中龙致,initialize的調(diào)用順序是什么了。
總結(jié):
1顷链,initialize的調(diào)用肯定與load的調(diào)用順序有關(guān)目代。
2,類別中如果沒有重寫initialize嗤练,則不會調(diào)用類中的initialize榛了。
3,各個類別中重寫initialize潭苞,互相沒有影響忽冻。一個類別中沒有重寫不會調(diào)用其他類別中已經(jīng)重寫的initialize。
4此疹,如果類和類別中重寫了initialize方法僧诚,運行程序時,只會有一個重寫的方法被運行時提前并自動運行蝗碎。
====>那么到底會運行哪一個重寫的initialize呢湖笨?<====
@1,通過重寫類和每個類別的load方法蹦骑,確定它們被引用的順序慈省。
@2,確定類和類別中重寫了initialize方法的眠菇。
@3边败,根據(jù)引用順序找出最后一個重寫了initialize的類別袱衷,則該類別重寫的initialize會被運行時自動運行,其他重寫的initialize不會被自動運行笑窜。
資料鏈接:
[IDER]:
http://blog.iderzheng.com
http://blog.iderzheng.com/objective-c-load-vs-initialize/
[MrPeak雜貨鋪]:http://blog.csdn.net/hanangellove/article/details/45033453
[知乎上的一個怎么面試iOS工程師的問題]:
http://blog.csdn.net/hanangellove/article/details/45033453