iOS之 Category 属性 的理解
在 Objective-C 中可以通过 Category 给一个现有的类添加属性,但是却不能添加实例变量
反正读第一遍的时候我是有点晕的,可以添加“属性”,然后又说“添加实例变量”,第一感觉就好像 有点自相矛盾了。那么我们谈谈:
- 什么是实例变量?
实例变量就是一种变量,可以存放数据的。在oc上,形式就如:
@interface MyViewController :UIViewController
{
UIButton *myButton;
} @end
- 什么是属性?
属性是oc的一个新的机制,并且要求你必须声明与之对应的实例变量,这是最早期的一种概念,随着编译器从GCC转换为LLVM(low level virtual machine),从此不再需要为属性声明实例变量了,如果LLVM发现一个没有匹配实例变量的属性,它将自动创建一个以下划线开头的实例变量。
- Category可以给一个现有的类添加属性
我建立了个 UIColor的hex分类 ,在UIColor+hex.h上我做了这么一件事:

然后做你会发现编译器并不会报任何错误,build一下也不会有问题,当然这不能说这段代码是可以运行的【crash】。仔细的话,你会发现在.m中是有警告的。

它说,属性 hex的setHex和hex方法需要定义使用 @dynamic或者自己实现。 这里添加的属性,不会自动生成实例变量,这里添加的属性其实是添加的getter与setter方法。
这里,我们是不是可以理解上面的那句话了,其实就是等同于这句话
类别中只能添加方法,不能添加实例变量
首先,我们就用了@dynamic去做:

警告确实立马消失,但是是不是可以运行了呢?梦想是美好的,结果依旧crash。
- 那么这个@dynamic到底干了什么可以让警告消除,当然你会不会发现这个@dynamic在哪里见过?对了在你用coredata的时候,你就会发现了【NSManagedObject自己探索去吧】。
@dynamic 是相对于 @synthesize的,它们用样用于修饰 @property,用于生成对应的的getter和setter方法。但是@ dynamic表示这个成员变量的getter和setter方法并不是直接由编译器生成,而是手工生成或者运行时生成。
@dynamic just tells the compiler that the getter and setter methods are implemented not by the class itself but somewhere else (like the superclass or will be provided at runtime).
那为什么会崩溃?哈哈,不要想当然了,因为你没有在runtime中提供getter and setter methods。那当然会崩溃啦。
上面我们就基本上了解了在category添加属性的一些问题。那么该如何解决这些问题,也是下面我想说的Associated Objects,let's go!
首先我们带着这么2个疑问:
- 关联对象被存放到什么地方,是不是存放在被关联对象本身的内存中?
- 关联对象的生命周期怎么样。什么时候被释放,什么时候被移除?


上面是两种方法的实现
- objc_setAssociatedObject
- objc_getAssociatedObject
- objc_removeAssociatedObjects
其中第三个函数objc_removeAssociatedObjects 函数我们一般式用不上的,因为这个函数会移除一个对象的所有关联对象,该将对象恢复为“
pristine state”。一般我我没问你如果要移除一个关联对象,我们会使用objc_setAssociatedObject 函数中传入nil就可以移除。

- 你会发现这两个函数都有一个key,在关联对象的时候我们使用了两种形式,那么这个key的设置有什么要求么?
key 值必须是一个对象级别的唯一常量
从上面的两种实现关联对象的方法中,我们使用了两种不同的key值
- static void *kAssociatedObjectKey = &kAssociatedObjectKey;
用 selector ,使用 getter 方法的名称作为 key 值。
- 其实还有一种 static char kAssociatedObjectKey; objc_getAssociatedObject(self, &kAssociatedObjectKey);
对比这三种方式,很多人都喜欢第二种
- 关于这个 objc_AssociationPolicy(关联策略)又是什么?
|
关联策略 |
等价属性 |
说明 |
|
OBJC_ASSOCIATION_ASSIGN |
@property (assign) or @property (unsafe_unretained) |
弱引用关联对象 |
|
OBJC_ASSOCIATION_RETAIN_NONATOMIC |
@property (strong, nonatomic) |
强引用关联对象,且为非原子操作 |
|
OBJC_ASSOCIATION_COPY_NONATOMIC |
@property (copy, nonatomic) |
复制关联对象,且为非原子操作 |
|
OBJC_ASSOCIATION_RETAIN |
@property (strong, atomic) |
强引用关联对象,且为原子操作 |
|
OBJC_ASSOCIATION_COPY |
@property (copy, atomic) |
复制关联对象,且为原子操作 |
上面这张表很清晰的告诉你这个关联策略的作用。
iOS之 Category 属性 的理解的更多相关文章
- iOS数据存储之属性列表理解
iOS数据存储之属性列表理解 数据存储简介 数据存储,即数据持久化,是指以何种方式保存应用程序的数据. 我的理解是,开发了一款应用之后,应用在内存中运行时会产生很多数据,这些数据在程序运行时和程序一起 ...
- IOS学习5——属性与成员变量
[转]iOS中属性与成员变量的区别 ios中属性修饰符的作用 1. 属性用property声明 2. 简而言之,对于目前的ios开发,属性和成员变量的区别,完全可以不管. 3. 这个是历史原因造成的. ...
- iOS开发-automaticallyAdjustsScrollViewInsets属性
iOS开发-automaticallyAdjustsScrollViewInsets属性 Available in iOS 7.0 and later. 简单点说就是automaticallyAdju ...
- iOS类别(Category)
iOS类别(Category)与扩展(Extension) 苹果的官方文档 Category在iOS开发中使用非常频繁.尤其是在为系统类进行拓展的时候,我们可以不用继承系统类,直接给系统类添加方法,最 ...
- IOS UITableView NSIndexPath属性讲解
IOS UITableView NSIndexPath属性讲解 查看UITableView的帮助文档我们会注意到UITableView有两个Delegate分别为:dataSource和deleg ...
- prototype属性的理解
1.对象:对象是JS的基本数据类型(原始类型(数字.字符串和布尔值),对象类型).对象是一种复合值:它将很多值(原始值或者其他对象)聚合在一起,可通过名字访问这些值. 2.三类JS对象和两类属性: 内 ...
- Intent的属性及Intent-filter配置——Action、Category属性与intent-filter属性
Intent的Action.Category属性都是一个普通的字符串,其中Action代表该Intent所要完成的一个抽象“动作”,而Category则用于为Action增加额外的附加列别的信息.通常 ...
- iOS开发 关于addChildViewController的理解
iOS开发 关于addChildViewController的理解 前言 我之前是做Android开发的接触ios开发不到一个月的时间,所以在有些东理解上会不自觉的向Android方向靠拢. 理解 通 ...
- Intent的Component,Action和Category属性详解-android学习之旅(五十)
Component属性 代码示例 public class MainActivity extends Activity{ @Override protected void onCreate(Bundl ...
随机推荐
- Spring - bean的lazy-init属性(懒加载)
默认情况下,容器初始化的时候便会把bean实例化,通常这样做可以让一些配置或者bean实例化的异常在容器启动的时候就发现,而不是在N久之后.但有时候,我们希望某个可能不会用到但又不是100%不用的be ...
- Python学习笔记012_网络_异常
1,Python如何访问互联网? url + lib = urllib >>> # 使用urllib包下的request模块 >>> >>> i ...
- java 如何判断操作系统是Linux还是Windows
String os = System.getProperty("os.name"); if(os.toLowerCase().startsWith("win") ...
- birt 报表设计总结
1, 通过sql查询出来的数据,当某个字段没有值时,我们期望显示别的东东 在表格单元格或者网格中选中这个值, 在属性编辑器-->映射--> 在映射表中添加映射条件 映射中当使用 等于 时, ...
- 关于使用lombok遇到的问题
在官网上下载了lombok.jar包以后,有两种安装方式 : 1. 双击下载下来的 JAR 包安装 lombok 我选择这种方式安装的时候提示没有发现任何 IDE,所以我没安装成功,我是手动安装 ...
- JQuery中关于浏览器兼容性的问题
前 言 LIUWE JQuery是一个特别强大的javascript代码库,,它的操作DOM的能力是相当强大的,JQuery可以说是支持各大主流浏览器,但是随着时代的不断发展,浏览器是在不断的更 ...
- GCD之异步同步体会
前面的博文也有写到同步异步,可能是看他人的博文,自己没有实验,感觉理解不深,所以就敲了些代码比较一下串行.并行分别对应的同步.异步. 1.首先创建串行.并行线程队列 1 2 dispatch_queu ...
- Java 编程思想 Chapter_14 类型信息
本章内容绕不开一个名词:RTTI(Run-time Type Identification) 运行时期的类型识别 知乎上有人推断作者是从C++中引入这个概念的,反正也无所谓,理解并能串联本章知识才是最 ...
- 分享ES6中比较常用又强大的新特性
前言 es6有很多新东西,但是感觉常用的并不是很多,这里学习记录了一些我自己认为非常常用又强大的新特性. scoping 实用的块级作用域,let x = xxx 可以声明一个块级作用域的局部变量,简 ...
- 基于java生成二维码
二维码 二维码的概念 ...