您的位置:首页 > 编程语言 > Go语言

Category 类扩展,就是给原有类的添加方法。

2016-06-29 23:12 302 查看
总结

1.分类中只能写方法的声明和实现,不要直接写属性,写了就报错

2.在分类的.h中用@property声明属性,不会生成属性,只是会在分类的.h中生成 “所谓的” _属性 的set/get声明.

3.在分类中访问本类中的属性

1>分类中不能直接访问本类中的真私有属性,但是可以通过set/get方法访问

@implementation YYNetworkTools (Download)

-(void)test

{

[self fileLength];

[self setFileLength:@”1.22”];

}

@end

2>分类中可以直接访问本类.h中直接声明的属性

@implementation YYNetworkTools (Download)

-(void)test

{

[self fileLength];

[self setFileLength:@”1.22”];

_text//可以访问

}

@end

一、作用

a) 将类的实现分开在几个不同的文件中

b) 声明私有方法

c) 模拟多继承

d) 把framework的私有方法公开

二、与extension的异处

extension是编译器决议,和类的头文件里的@interface以及实现文件里面的@implement一起形成一个完整的类。你必须用一个类的源码才能为它extension,比如你无法为NSString添加extension。

category是运行期决议的。

所以extension可以添加实例变量,而category作为运行期决议是无法添加实例变量,因为会破坏类的内存布局。

三、运行时的category

众所周知,运行时,OC几乎所有的内容都是以结构体的形式存在的。catrgory也不例外。

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;

1)、类的名字(name)

2)、类(cls)

3)、category中所有给类添加的实例方法的列表(instanceMethods)

4)、category中所有添加的类方法的列表(classMethods)

5)、category实现的所有协议的列表(protocols)

6)、category中添加的所有属性(instanceProperties)

原文从详细介绍了category的方法是如何添加至类的方法列表中。

而且得出结论如下:

category并不是绝对的覆盖了类的同名方法,而是catrgory的方法被排在了类的同名方法之前,而方法的检索方式是顺序检索,所以在调用方法时,调用到的同名方法是category的,进而产生了覆盖效果。

四、进阶

一个类的两个category,如果存在方法名相同,是根据buildPhases->Compile Sources里面的从上至下顺序编译的,即后编译的会被调用。

在Xcode->Edit Scheme->Run->Arguments->Environment Variables中添加运行时环境变量可打印相应category加载消息。

(更多变量尽在objc-private.h中)

下列代码是利用运行时遍历方法列表,调用被category覆盖的方法:

Class currentClass = [VVStack class];
VVStack * my = [VVStack new];
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
NSString * methodName = [NSString stringWithCString:sel_getName(method_getName(method)) encoding:NSUTF8StringEncoding];
if ([@"printName" isEqualToString:methodName]) {
//得到方法对应的函数的指针
lastImp = method_getImplementation(method);
//得到方法的名字 SEL类型
lastSel = method_getName(method);
}
}

typedef void (*functionn) (id, SEL);
if (lastImp!= NULL) {
functionn f = (functionn)lastImp;
//传入类名/实例名 、方法的SEL、对应参数 开始调用方法
f(my,lastSel);
}
//释放方法列表
free(methodList);
}


ab49
虽说category中无法直接添加属性,但是我们在Category中可以利用运行时添加属性:

(void)setName:(NSString *)name

{

objc_setAssociatedObject(self,

“name”,

name,

OBJC_ASSOCIATION_COPY);

}

(NSString*)name

{

NSString *nameObject = objc_getAssociatedObject(self, “name”);

return nameObject;

}

如此便可以在类中利用点语法调用name属性。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: