在 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个疑问:

  1. 关联对象被存放到什么地方,是不是存放在被关联对象本身的内存中?
  2. 关联对象的生命周期怎么样。什么时候被释放,什么时候被移除?

上面是两种方法的实现

  • objc_setAssociatedObject
  • objc_getAssociatedObject
  • objc_removeAssociatedObjects

其中第三个函数objc_removeAssociatedObjects 函数我们一般式用不上的,因为这个函数会移除一个对象的所有关联对象,该将对象恢复为“

pristine state”。一般我我没问你如果要移除一个关联对象,我们会使用objc_setAssociatedObject 函数中传入nil就可以移除。

  • 你会发现这两个函数都有一个key,在关联对象的时候我们使用了两种形式,那么这个key的设置有什么要求么?

  key 值必须是一个对象级别的唯一常量

从上面的两种实现关联对象的方法中,我们使用了两种不同的key值

  1. static void *kAssociatedObjectKey = &kAssociatedObjectKey;
  2. 用 selector ,使用 getter 方法的名称作为 key 值。

  3. 其实还有一种 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)

复制关联对象,且为原子操作

  上面这张表很清晰的告诉你这个关联策略的作用。

  1. 关联对象与被关联对象本身的存储并没有直接的关系,它是存储在单独的哈希表中的;
  2. 关联对象的五种关联策略与属性的限定符非常类似,在绝大多数情况下,我们都会使用 OBJC_ASSOCIATION_RETAIN_NONATOMIC 的关联策略,这可以保证我们持有关联对象;
  3. 关联对象的释放时机与移除时机并不总是一致,比如实验中用关联策略 OBJC_ASSOCIATION_ASSIGN 进行关联的对象,很早就已经被释放了,但是并没有被移除,而再使用这个关联对象时就会造成 Crash 。

iOS之 Category 属性 的理解的更多相关文章

  1. iOS数据存储之属性列表理解

    iOS数据存储之属性列表理解 数据存储简介 数据存储,即数据持久化,是指以何种方式保存应用程序的数据. 我的理解是,开发了一款应用之后,应用在内存中运行时会产生很多数据,这些数据在程序运行时和程序一起 ...

  2. IOS学习5——属性与成员变量

    [转]iOS中属性与成员变量的区别 ios中属性修饰符的作用 1. 属性用property声明 2. 简而言之,对于目前的ios开发,属性和成员变量的区别,完全可以不管. 3. 这个是历史原因造成的. ...

  3. iOS开发-automaticallyAdjustsScrollViewInsets属性

    iOS开发-automaticallyAdjustsScrollViewInsets属性 Available in iOS 7.0 and later. 简单点说就是automaticallyAdju ...

  4. iOS类别(Category)

    iOS类别(Category)与扩展(Extension) 苹果的官方文档 Category在iOS开发中使用非常频繁.尤其是在为系统类进行拓展的时候,我们可以不用继承系统类,直接给系统类添加方法,最 ...

  5. IOS UITableView NSIndexPath属性讲解

    IOS UITableView NSIndexPath属性讲解   查看UITableView的帮助文档我们会注意到UITableView有两个Delegate分别为:dataSource和deleg ...

  6. prototype属性的理解

    1.对象:对象是JS的基本数据类型(原始类型(数字.字符串和布尔值),对象类型).对象是一种复合值:它将很多值(原始值或者其他对象)聚合在一起,可通过名字访问这些值. 2.三类JS对象和两类属性: 内 ...

  7. Intent的属性及Intent-filter配置——Action、Category属性与intent-filter属性

    Intent的Action.Category属性都是一个普通的字符串,其中Action代表该Intent所要完成的一个抽象“动作”,而Category则用于为Action增加额外的附加列别的信息.通常 ...

  8. iOS开发 关于addChildViewController的理解

    iOS开发 关于addChildViewController的理解 前言 我之前是做Android开发的接触ios开发不到一个月的时间,所以在有些东理解上会不自觉的向Android方向靠拢. 理解 通 ...

  9. Intent的Component,Action和Category属性详解-android学习之旅(五十)

    Component属性 代码示例 public class MainActivity extends Activity{ @Override protected void onCreate(Bundl ...

随机推荐

  1. eclipse设置背景保护色及常用设置

    1.设置背景颜色 2.代码自动补全 Windows-->Preferences-->Java-->Editor-->Content Asist,在Auto activation ...

  2. Linux下安全证书申请以及配置到Nginx

    wget https://raw.githubusercontent.com/xdtianyu/scripts/master/lets-encrypt/letsencrypt.shchmod +x l ...

  3. Nodejs最好的ORM - TypeORM

    TypeORM是一个采用TypeScript编写的用于Node.js的优秀ORM框架,支持使用TypeScript或Javascript(ES5, ES6, ES7)开发.目标是保持支持最新的Java ...

  4. java 基础语法 2

    一.语句

  5. Apache Spark 2.2.0 中文文档 - Submitting Applications | ApacheCN

    Submitting Applications 在 script in Spark的 bin 目录中的spark-submit 脚本用与在集群上启动应用程序.它可以通过一个统一的接口使用所有 Spar ...

  6. C++ static成员变量与static成员函数

    类中的静态成员真是个让人爱恨交加的特性.我决定好好总结一下静态类成员的知识点,以便自己在以后面试中,在此类问题上不在被动. 静态类成员包括静态数据成员和静态函数成员两部分.  一 静态数据成员:  类 ...

  7. hive参数配置

    CLI参数 两种修改方式: 1)启动时 hive --hiveconf  hive.cli.print.current.db=true 2)修改当前用户home目录下 .hiverc文件,hive c ...

  8. 【转】常用Maven插件

    我们都知道Maven本质上是一个插件框架,它的核心并不执行任何具体的构建任务,所有这些任务都交给插件来完成,例如编译源代码是由maven- compiler-plugin完成的.进一步说,每个任务对应 ...

  9. JSP入门3 Servlet

    需要继承类HttpServlet 服务器在获得请求的时候会先根据jsp页面生成一个java文件,然后使用jdk的编译器将此文件编译,最后运行得到的class文件处理用户的请求返回响应.如果再有请求访问 ...

  10. hdu4678 Mine 2013 Multi-University Training Contest 8 博弈题

    Mine Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65535/32768 K (Java/Others) Total Submi ...