分类

常见问题

分类的加载。。load/initial的执行

创建Person 类、创建student类继承person、为student添加分类

1、在类的+load方法调用的时候,我们可以调用category中声明的方法么?
2、+load方法,调用顺序是咋样的呢?
3、initial方法,调用顺序是咋样的呢?

load方法。类加载的时候执行一次。不受分类影响。。顺序是父类、本类、分类。。
分类的load顺序是按照编译顺序compile Sources内从上到下。。
initial和普通方法一样。收到分类加载影响。。 执行顺序是父类、本类(如果分类有覆盖,则执行覆盖的分类方法。且执行按照编译顺序的最后一个分类的方法。。)
那如何调用到其它被”覆盖“的方法呢?需要直接遍历查找。。

Class currentClass = [MyClass class];
MyClass *my = [[MyClass alloc] init];

if (currentClass) {
    unsigned int methodCount;
    Method *methodList = class_copyMethodList(currentClass, &methodCount);
    IMP lastImp = NULL;
    SEL lastSel = NULL;
    for (NSInteger i = 0; i < methodCount; i++) {
        Method method = methodList[i];
        NSString *methodName = [NSString stringWithCString:sel_getName(method_getName(method)) 
                                        encoding:NSUTF8StringEncoding];
        if ([@"printName" isEqualToString:methodName]) {
            lastImp = method_getImplementation(method);
            lastSel = method_getName(method);
        }
    }
    typedef void (*fn)(id,SEL);
    
    if (lastImp != NULL) {
        fn f = (fn)lastImp;
        f(my,lastSel);
    }
    free(methodList);
}   

执行顺序举例

2021-06-23 14:18:35.868867+0800 NetworkDemo[59483:6920152] +[Person load]2021-06-23 14:18:35.873671+0800 NetworkDemo[59483:6920152] +[Stuent load]2021-06-23 14:18:35.883234+0800 
NetworkDemo[59483:6920152] +[Stuent(Add1) load]2021-06-23 14:18:35.883449+0800 
NetworkDemo[59483:6920152] +[Stuent(Add2) load]2021-06-23 14:18:36.498388+0800 
NetworkDemo[59483:6920152] -[SingleDemo mutableCopyWithZone:]2021-06-23 14:18:36.498554+0800 
NetworkDemo[59483:6920152] +[Person initialize]2021-06-23 14:18:36.498691+0800 
NetworkDemo[59483:6920152] +[Stuent(Add2) initialize]2021-06-23 14:18:36.498840+0800 
NetworkDemo[59483:6920152] -[Stuent(Add2) instanceMethod]2021-06-23 14:18:36.498975+0800 
NetworkDemo[59483:6920152] +[Stuent(Add2) classMethod]

如果本类、分类都没有initial方法 则父类执行两次,这是因为消息传递机制找不到本类initial方法直接去调用了父类的

2021-06-23 14:15:35.259318+0800 NetworkDemo[59436:6916267] +[Person load]2021-06-23 
14:15:35.260071+0800 NetworkDemo[59436:6916267] +[Stuent load]2021-06-23 14:15:35.268680+0800 
NetworkDemo[59436:6916267] +[Stuent(Add1) load]2021-06-23 14:15:35.268795+0800 
NetworkDemo[59436:6916267] +[Stuent(Add2) load]2021-06-23 14:15:36.727885+0800 
NetworkDemo[59436:6916267] +[Person initialize]2021-06-23 14:15:36.728203+0800 
NetworkDemo[59436:6916267] +[Person initialize]2021-06-23 14:15:36.728349+0800 
NetworkDemo[59436:6916267] -[Stuent(Add2) instanceMethod]2021-06-23 14:15:36.728468+0800 
NetworkDemo[59436:6916267] +[Stuent(Add2) classMethod]


分类是否可以添加属性

属性、成员变量

@interface Stuent ()

@property(nonatomic,assign) int  age; //属性

@property(nonatomic,copy) NSString * name;// 属性

@end

@implementation Stuent
{
   NSString* tempName;//成员变量
}

也可以直接runtime直接获取看下
     unsigned int ivarCount = 0;     
Ivar *ivars = class_copyIvarList([Stuent class], &ivarCount);     
for (int i = 0; i < ivarCount; i++) {          
    Ivar ivar = ivars[i];          
    NSLog(@"成员变量:%s",ivar_getName(ivar));      
  }      
  free(ivars);
//  属性     
unsigned int propertyCount = 0;      
objc_property_t *propertyList = class_copyPropertyList([Stuent class], &propertyCount);     
for (int i = 0; i < propertyCount; i++) {         
objc_property_t property = propertyList[i];         
NSLog(@" 属性:%s" ,property_getName(property));     
}                 
//  方法列表      
unsigned int methodCount = 0;     
Method *methods = class_copyMethodList([Stuent class], &methodCount);     
for (int i = 0; i < methodCount; i++) {         
Method method = methods[i];                    
NSLog(@" 方法:%s" , sel_getName(method_getName(method)));      
}

属性== 相应的下划线成员变量+set、get方法,分类中。我添加属性。。默认就不会生成成员变量以及setget方法。。即便你手动实现。。因为成员变量缺失也不起作用。。。一般我们用关联方发

分类中添加
@property(nonatomic,copy) NSString * name;
-(NSString)name{returnobjc_getAssociatedObject(self,@"name");}------(void)setHeight(NSString)height{objc_setAssociatedObject(self,@"name",height,OBJC_ASSOCIATION_COPY_NONATOMIC);}@end
底层原因是因为category_t结构体中并不存在成员变量。分类是运行时才去加载的;通过之前对对象的分析我们知道成员变量是存放在实例对象中的,并且编译的那一刻就已经决定好了。而分类是在运行时才去加载的。那么我们就无法再程序运行时将分类的成员变量中添加到实例对象的结构体中。因此分类中不可以添加成员变量。

category和extension

extension在编译期决议,它就是类的一部分,在编译期和头文件里的@interface以及实现文件里的@implement一起形成一个完整的类,它伴随类的产生而产生,亦随之一起消亡。extension一般用来隐藏类的私有信息,你必须有一个类的源码才能为一个类添加extension,所以你无法为系统的类比如NSString添加extension。
但是category则完全不一样,它是在运行期决议的。 就category和extension的区别来看,我们可以推导出一个明显的事实,extension可以添加实例变量,而category是无法添加实例变量的(因为在运行期,对象的内存布局已经确定)。
category底层:
typedef struct category_t {
const char *name;
classref_t cls;
struct method_list_t *instanceMethods;
struct method_list_t *classMethods;
struct protocol_list_t *protocols;
struct property_list_t *instanceProperties;
} category_t;
从category的定义也可以看出category的可为(可以添加实例方法,类方法,甚至可以实现协议,添加属性)和不可为(无法添加实例变量)。
当一个类有自己的一个分类,类目底层的的变化是:
1)、首先编译器生成了实例方法列表OBJC_CATEGORY_INSTANCE_METHODSMyClass_MyAddition和属性列表OBJC_PROP_LISTMyClass_MyAddition,两者的命名都遵循了公共前缀+类名+category名字的命名方式,而且实例方法列表里面填充的正是AdditionCategory里面的方法,而属性列表里面填充的也正是分类里添加的属性。
还有一个需要注意到的事实就是category的名字用来给各种列表以及后面的category结构体本身命名,而且有static来修饰,所以在同一个编译单元里我们的category名不能重复,否则会出现编译错误。
2)、其次,编译器生成了category本身OBJC_CATEGORYMyClass_MyAddition,并用前面生成的列表来初始化category本身。
3)、最后,编译器在DATA段下的objc_catlist section里保存了一个大小为1的category_t的数组L_OBJC_LABELCATEGORY$(当然,如果有多个category,会生成对应长度的数组),用于运行期category的加载。

1)、把category的实例方法、协议以及属性添加到类上
2)、把category的类方法和协议添加到类的metaclass上

!1、category的方法没有“完全替换掉”原来类已经有的方法,也就是说如果category和原来类都有methodA,那么category附加完成之后,类的方法列表里会有两个methodA

!2、category的方法被放到了新方法列表的前面,而原来类的方法被放到了新方法列表的后面,运行时在查找方法的时候是顺着方法列表的顺序查找的,优先找到最前面的进行调用

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,640评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,254评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,011评论 0 355
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,755评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,774评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,610评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,352评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,257评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,717评论 1 315
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,894评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,021评论 1 350
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,735评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,354评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,936评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,054评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,224评论 3 371
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,974评论 2 355

推荐阅读更多精彩内容