OC为啥 Category不能添加属性

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了OC为啥 Category不能添加属性相关的知识,希望对你有一定的参考价值。

参考技术A 用Clang编译我的.m文件【clang -rewrite-objc MyObject.m】会生成.cpp文件

编译器把我的对象MyObject转成了下面的结构体
MyObject_IMPL

NSObject_IMPL

isa 是 Class类型, Class是objc_class类型的指针

extension看起来很像一个匿名的category,但是extension和有名字的category几乎完全是两个东西。 extension在编译期决议,它就是类的一部分,在编译期和头文件里的@interface以及实现文件里的@implement一起形成一个完整的类,它伴随类的产生而产生,亦随之一起消亡。extension一般用来隐藏类的私有信息,你必须有一个类的源码才能为一个类添加extension,所以你无法为系统的类比如NSString添加extension。

但是category则完全不一样,它是在运行期决议的。
就category和extension的区别来看,我们可以推导出一个明显的事实,extension可以添加实例变量,而category是无法添加实例变量的(因为在运行期,对象的内存布局已经确定,如果添加实例变量就会破坏类的内部布局,这对编译型语言来说是灾难性的)。

为什么说是灾难性的?考虑下已经存在于内存中的这类对象该怎么处理?

那objc_setAssociatedObject又是如何实现的呢?下一篇寻求答案

iOS的Runtime机制下给类别(category)添加属性替换原有类的方法执行

一、Runtime的理解

  OC是面向对象的语言这是常识,其实就是通过Runtime机制动态创建类和对象,这里只是简单的运用runtime的使用!

 

二、类别(category)添加属性_使用前记得导入头文件#import <objc/runtime.h>

  通过这三个方法动态的绑定属性

OBJC_EXPORT void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy)
    OBJC_AVAILABLE(10.6, 3.1, 9.0, 1.0);//set方法
OBJC_EXPORT
id objc_getAssociatedObject(id object, const void *key) OBJC_AVAILABLE(10.6, 3.1, 9.0, 1.0);//get方法
OBJC_EXPORT
void objc_removeAssociatedObjects(id object) OBJC_AVAILABLE(10.6, 3.1, 9.0, 1.0);//移除

  eg:

- (void)setName:(NSString *)name{
    
//    OBJC_ASSOCIATION_ASSIGN = 0,           //assign
//    OBJC_ASSOCIATION_RETAIN_NONATOMIC = 1, //retaion , nonatomic
//    OBJC_ASSOCIATION_COPY_NONATOMIC = 3,   //copy ,  nonatomic
//    OBJC_ASSOCIATION_RETAIN = 01401,       //retain
//    OBJC_ASSOCIATION_COPY = 01403          //copy
   /**
    实现set方法

    @param object#> self description#>
    @param key#> 属性的键值统一就行 description#>
    @param value#> 属性的值 description#>
    @param policy#> 和copy,retain,assign类似 description#>
    */
    objc_setAssociatedObject(self, nameKey, name, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

//定义常量 属性的键值统一
static char  *nameKey = "nameKey";

- (NSString *)name{
    /**
     实现get方法

     @param object#> 这是填self description#>
     @param key#> 属性的键值统一就行 description#>
     */
    return objc_getAssociatedObject(self, nameKey);

 

三、两个方法的执行替换

    SEL originalSelector = NSSelectorFromString(@"原来方法名");
    SEL swizzledSelector = NSSelectorFromString(@"需要替换的方法名");
    Method originalMethod = class_getInstanceMethod([self class], originalSelector);
    Method swizzledMethod = class_getInstanceMethod([self class], swizzledSelector);
    method_exchangeImplementations(originalMethod, swizzledMethod);

 

 

 

 

 

 



以上是关于OC为啥 Category不能添加属性的主要内容,如果未能解决你的问题,请参考以下文章

OC-category 为什么不能添加成员变量

对OC中Category 的理解,以及方法覆盖的解决办法

iOS开发之category(类别)

给分类(Category)添加属性

OC-类目延展协议

OC category如何注入新的属性(property)