一概述:
Category是Objective-C 2.0之后添加的语言特性,Category又叫分类、类别、类目,能够在不改变原来类内容的基础上,为类增加一些方法。
除此之外,Category还有以下功能:
(1)将类的实现分开写在几个分类里面。
这样做的好处:
- 可以减少单个文件的体积
- 可以把不同的功能组织到不同的Category里
- 可以由多个开发者共同完成一个类
- 可以按需加载想要的category
(2)声明私有的方法。
(3)模拟多继承。
二.Category的定义与使用
例如,我们创建一个Person类,并为其创建一个Category命名为MyCategory。创建Category很简单,如下图:
为Person创建一个名为MyCategory的Category后,会自动生成Person+MyCategory.h和Person+MyCategory.m文件。
我们在MyCategory中声明和实现一个read方法,如下:
#import "Person.h"
@interface Person (MyCategory)
-(void)read;
@end
#import "Person+MyCategory.h"
@implementation Person (MyCategory)
-(void)read{
NSLog(@"调用了MyCategory的read方法!");
}
@end
int main(){
Person *p = [[Person alloc] init];
[p read];
}
打印结果:
调用了MyCategory的read方法!
注意:
- 分类只能增加方法,不能增加成员变量。
- 分类方法实现中可以访问原来类中声明的成员变量。
- 分类可以重新实现原来类中的方法,但是会覆盖掉原来的方法,会导致原来的方法没法再使用(实际上并没有真的替换,而是Category的方法被放到了新方法列表的前面,而原来类的方法被放到了新方法列表的后面,这也就是我们平常所说的Category的方法会“覆盖”掉原来类的同名方法,这是因为运行时在查找方法的时候是顺着方法列表的顺序查找的,它只要一找到对应名字的方法,就会罢休,殊不知后面可能还有一样名字的方法)。
- 当分类、原来类、原来类的父类中有相同方法时,方法调用的优先级:分类(最后参与编译的分类优先) –> 原来类 –> 父类,即先去调用分类中的方法,分类中没这个方法再去原来类中找,原来类中没有再去父类中找。
- Category是在runtime时候加载,而不是在编译的时候。
Category与成员变量、属性(以后再说。)
三.Extension的基本用法 :
Extension的创建方法与Category一样,只要在原来选择Category选择Extension即可,比如我们为Person创建一个名为MyExtension的Extension,则最终会生成一个Person_MyExtension.h文件:
#import "Person.h"
@interface Person ()
@end
但要注意的是和Category不同的是它不会生成Person_MyExtension.m文件。之后我们可以在Person_MyExtension.h中直接添加成员变量、属性和方法,如下:
#import "Person.h"
@interface Person ()
{
NSString * _address;
}
@property (nonatomic) NSInteger age;
-(NSString*)WhereAmI;
@end
他常用的形式不是创建一个单独的文件,而是在实现文件中添加私有的成员变量、属性和方法。例如:
#import "Person.h"
/Extension start///
@interface Person ()
{
NSString * _address;
}
@property (nonatomic) NSInteger age;
-(NSString*)WhereAmI;
@end
/Extension end///
@implementation Person
-(NSString*)WhereAmI{
return @"谁知道你在哪里";
}
@end
Extension与Category区别:
- Extension
- 在编译器决议,是类的一部分,在编译器和头文件的@interface和实现文件里的@implement一起形成了一个完整的类。
- 伴随着类的产生而产生,也随着类的消失而消失。
- Extension一般用来隐藏类的私有消息,你必须有一个类的源码才能添加一个类的Extension,所以对于系统一些类,如NSString,就无法添加类扩展
- Category
- 是运行期决议的
- 类扩展可以添加实例变量,分类不能添加实例变量
- 原因:因为在运行期,对象的内存布局已经确定,如果添加实例变量会破坏类的内部布局,这对编译性语言是灾难性的。