iOS 探讨category和extension

一、分类(Category)

想必大家都知道我们可以在分类中为某个类扩展方法,并且通过分类添加的方法的优先级还高于原始类中的方法,即如果分类中和原始类中有相同的方法,那我们调用时是优先调用通过分类添加的方法。
既然通过分类能给类添加方法,那能不能给类添加属性呢?属性中绑定了一个成员变量,那分类又能不能给类添加成员变量呢?今天元宵节,闲着没事,就此调研了一番。

论点

(1) 分类可以给类添加属性
(2) 分类不可以给一个类扩展成员变量的`

先唠唠嗑

强调一下,分类不是类,它只是一个类似匿名扩展(下面会讨论扩展)的模块,用于扩展给类添加方法,便于协作、分模块的开发,你看看苹果提供的很多API都是这样,苹果会写很多分类来给某个类添加不同的功能,一来结构清晰,二来也便于协同开发,可以每个人负责一个分类。

我们都知道,OC的世界里,任何集成自NSObject的对象都有isa指针,他本质是一个结构体指针,但是对于Category来说,他是没有isa指针的,好了,下面进行论证一番。

论据

先说说属性,属性是对成员变量的一个封装,当我们声明一个属性的时候,Xcode会给我们默认创建一个 _属性名 命名的成员变量,也会给我们自动创建getter和setter方法。当然我们也可以用@synthesize指定其关联的变量

【例如】给属性name指定其关联的变量@synthesize name = xxx;执行self.name的时候,其实是操作的是实例变量xxx,而不是_name了。

窝草,扯远了,回归正传。。

所以我们要添加一个属性得有三样东西,settergetter以及关联的实例变量

在分类中Xcode不会为我们自动创建setter、getter方法,但是我们可以手动实现,但是如何把一个变量关联到属性上呢,直接声明一个全局变量不行吗,事实证明不太行,对于getter方法还好说,直接返回一个变量就行,可是setter方法却不行,因为你要找到该属性关联的变量你才能给人家赋值啊,怎么办?怎么办??经过调研发现这只能用runtime的对象关联来实现了

.m文件中
// 定义关联的key
static const char *key = "name";

@implementation NSObject (Property)

-  (NSString *)name {
  // 根据关联的key,获取关联的值。
  return objc_getAssociatedObject(self, key);
}

-  (void)setName:(NSString *)name {
  // 参数一:目标对象
  // 参数二:关联的key,可以通过这个key获取
  // 参数三:关联的value
  // 参数四:关联的策略
  objc_setAssociatedObject(self, key, name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@end

看到这里你也许就会说,这不是把变量给添加上去了吗?我表示默默一笑,如果你在原始类中打印IVarList你就会发现并没有这个变量,只能打印出添加的属性。
其实对象关联只是把属性关联上去了,并没有把变量添加进去,说了这么多这回你该信了吧。

原因就是分类不是类,他没有isa指针,下面是isa指针的代码,可以看出他本质上是一个objc_class结构体,通过isa指针才能找指向装有变量的ivars容器,也就是说你都不知道装有变量的容器ivars,你怎么去给它生猴子啊,但是奇怪了,ISA指针里没有指向属性数组的指针,没有是对的,要不然这一段的解释就废了,可是讲真,属性指针在哪啊???有知道的小伙伴请告诉我一下 >_< !!!

struct objc_class {
    Class isa  OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
    Class super_class                                        OBJC2_UNAVAILABLE;
    const char *name                                         OBJC2_UNAVAILABLE;
    long version                                             OBJC2_UNAVAILABLE;
    long info                                                OBJC2_UNAVAILABLE;
    long instance_size                                       OBJC2_UNAVAILABLE;
    struct objc_ivar_list *ivars                             OBJC2_UNAVAILABLE;
    struct objc_method_list **methodLists                    OBJC2_UNAVAILABLE;
    struct objc_cache *cache                                 OBJC2_UNAVAILABLE;
    struct objc_protocol_list *protocols                     OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;
/* Use `Class` instead of `struct objc_class *` */

结论

分类中可以给一个对象添加属性,但是不能添加实例变量,只能通过运行时关联上去。
注意:如果不信,你可以在原始类中打印一遍吧,会发现只能打印出添加的属性,打印不出变量。

二、扩展(Extension)

在分类中,我们可以添加属性,声明方法,但是这些都是私有的,因为你无法把扩展给导出去啊,这就是和category的一个区别。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 把网上的一些结合自己面试时遇到的面试题总结了一下,以后有新的还会再加进来。 1. OC 的理解与特性 OC 作为一...
    AlaricMurray阅读 2,592评论 0 20
  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,758评论 0 9
  • Getting started(入口)Welcome to Training for Android develo...
    AiPuff阅读 217评论 0 0
  • 或许是夏天快要来了的原因,天气变得让人感到异常闷热,写作业写得特别烦躁,根本写不进去,效率很低。为了缓解下心...
    7e阅读 451评论 0 3
  • 亲爱的妈妈: 昨天,我得知你送咪贝上编程课的时候和周老师聊了会天。你告诉周老师,惊喜地看到自己的女儿越来越出色,连...
    葛瑛阅读 217评论 2 6