一.分类(Category)
1.基本用途
不改变原来类模型(不修改原来类的任何内容)的前提下,想给类扩充一些方法,有2种:
(1)继承(间接扩充,通过继承获得父类的方法)
(2)分类(Category)(直接扩充方法,比较自由)
2.格式
(1)分类的声明
@interface 类名 (分类名称)
方法的声明
@end
就是把类后面的继承去掉,加上(),类名写在括号里。
(2)分类的实现
@implementation 类名 (分类名称)
方法的实现
@end
就是在类实现后面加上(),类名写在括号里
3.好处
(1)一个庞大的类可以分模块开发,某一类功能可以放在一个分类里,方便开发。
(2)庞大的类可以由多个人来编写,更有利于团队合作。
4.注意:
(1)分类实现中可以访问原来类里面的成员变量,但是不能添加成员变量,分类就是为了添加方法的。如果想添加成员变量要么去原来类里添加,要么用继承添加方法和成员变量。
(2)分类里面可以写跟原始类一样的方法,但不推荐这么做,因为它是直接替换掉原来的方法,因为调用方法时候,分类的优先级别最高,这么做的后果是再也不能访问原来的方法。
(3)如果多个分类写了同一个方法,只有最后一个参与编译的才有有效,就是最后一个覆盖了前面所有的同名方法。
(4)方法调用的优先级:分类(最后参与编译的优先)-->原来的类-->父类。
(5)大多数情况下,分类是给系统自带的类添加分类的,系统自带的类是不能修改的,只有通过这个方法才能增加系统
类的方法。
二.类的本质
1.类也是对象是Class类型的对象,简称“类对象(也就是类)”。
2.Class类型的定义:typedef struct objc_class *Class,利用Class创建Person类对象,然后利用Person类对象,创
建Person类型的对象。
3.类名就代表着类对象,每个类只有一个类对象(对象可以无限创建,类只有1个)。
4.+load
(1)在程序启动的时候会加载所有的类和分类,而且加载后会调用所有类和分类的+load方法,只会调用1次。
(2)先加载父类,再加载子类;也就是先调用父类的+load,再调用子类的+load
(3)先加载元原始类,再加载分类
(4)不管程序运行过程有没有用到这个类,都会调用+load加载
5.+initialize
(1) 在第一次使用某个类时(比如创建对象等),就会调用一次+initialize方法
(2)一个类只会调用一次+initialize方法,先调用父类的,再调用子类的
6.获取类对象的方法
[对象 class] :用对象的class方法,返回值是类对象,Class类型的指针存放。
[类 class]:用类的class方法,返回值是类对象,Class类型的指针存放。
既然返回的是类对象(就是类),就可以使用这个类对象的类方法,比如:new,alloc等。
三.description方法
1.-description方法
使用NSLog和%@直接输出某个对象时,会调用对象的-description方法,并拿到返回值(NSString *)进行输出。默认会
输出<类名:对象地址>。可以重写-description方法,输出自己想输出的内容。
2.+description方法
使用NSLog和%@直接输出某个类对象时,会调用类对象+description方法,并拿到返回值(NSString *)输出。默认只会
输出类名。
3.如果想修改NSLog的默认输出可以重写-description方法和+description方法。
4.如果在description方法中使用NSLog输出self,会陷入死循环。
5.NSLog输出C语言字符串的时候不能有中文。
四.SEL(消息)
SEL其实是对方法的一种包装,将方法包装成一个SEL类型的数据,去找对应的方法地址。找到方法地址就可以调用方法。调用方法就是先把方法包装成SEL类型数据-->根据SEL数据找到对应的方法的地址-->根据方法地址调用方法。
1.方法的存储位置
(1)每个类的方法列表都存储在类对象中。
(2)每个方法都有一个与之对应的SEL类型的对象。
(3)根据一个SEL对象就可以找到方法的地址,进而调用方法。
(4)SEL类型的定义:typedef struct objc_selector *SEL 。
2.SEL对象的创建
(1)SEL s =@selector(方法名):用@selector把方法名转变成SEL类型的对象,这样就可以找到方法的地址,进而调用方法。
(2)SEL s = NSSelectorFromString(@"test");用函数将字符串型的方法名转变成SEL类型的对象。
3.SEL对象的其他用法
(1)可以用函数NSStringFromSelector将SEL对象转变成NSString对象(字符串)。
(2)调用方法可以有另外一种方式:[对象 performSelector:SEL对象]或[对象 performSelector:SEL对象 withObject:参数]。
(3)每个方法里面都有一个SEL数据代表当前方法:_cmd。
五.NSLog输出增强
下面的那些宏前面都是2个下划线
1.__FILE__ :源代码文件的路径(%s-必须用printf函数输出)
2.__LINE__ :NSLog代码在第几行(%d)
3.__func__ :函数名(%s)
4. _cmd :代表着当前方法的SEL
5.如果在方法中[self performSelector:_cmd]会导致死循环。