一切皆为对象。
其实有很多种方式你可以在面向过程和面向对象间互相转化,但本文的目的是:有时候抛弃C层面的东西也是很好的。
是的——对于这种Smalltalk一样杂交而成的语言中的非面向对象部分而言,C语言是很有魅力的一部分。它速度快、久经考验,是现代计算最核心的部分。而且当面向对象范式处于过于庞大的设计而显得臃肿不堪的时候,C就变成了你的“安全出口”。
静态函数比硬要塞入类中的方法要好。 枚举比字符串常量要好。 按位掩码比字符串常量组成的数组要好。 过程化指令比runtime hack要好。
一个专业的Objective-C工程师应该在面向对象和面向过程范式间优雅地切换,同时能够掌握二者的优势。
而关于这一点,本周的话题围绕的是这两个简单而方便的宏定义:
1 | NS_ENUM |
和
1 | NS_OPTIONS |
。
1 | NS_ENUM |
和
1 | NS_OPTIONS |
都不算太古老的宏,在iOS 6 / OS X Mountain Lion才开始有,他们都是代替
1 | enum |
的更好的办法。
如果你想在更早的iOS或OS X系统中使用这两个宏,简单定义一下就好了:
1
<span class="cp">#ifndef NS_ENUM</span> <span class="cp">#define NS_ENUM(_type, _name) enum _name : _type _name; enum _name : _type</span> <span class="cp">#endif</span>
1 | enum |
,或者其他枚举类型(例如每周的星期几,或TableViewCell的类型等),都是通过C的方法去为预设值定义常量。在一个
1 | enum |
定义中,没有被赋予特别值的常量都会自动被赋为从0开始的连续值。
有几种合法的方式来定义
1 | enum |
。容易产生困惑的地方是它们每种方法之间略有不同,但不必须想太多,任选一种即可。
例如:
1
2
3
4
5
6
<span class="k">enum</span> <span class="p">{</span>
<span class="n">UITableViewCellStyleDefault</span><span class="p">,</span>
<span class="n">UITableViewCellStyleValue1</span><span class="p">,</span>
<span class="n">UITableViewCellStyleValue2</span><span class="p">,</span>
<span class="n">UITableViewCellStyleSubtitle</span>
<span class="p">};</span>
…定义整型值,但不定义类型。
另一种方法:
1
2
3
4
5
6
<span class="k">typedef</span> <span class="k">enum</span> <span class="p">{</span>
<span class="n">UITableViewCellStyleDefault</span><span class="p">,</span>
<span class="n">UITableViewCellStyleValue1</span><span class="p">,</span>
<span class="n">UITableViewCellStyleValue2</span><span class="p">,</span>
<span class="n">UITableViewCellStyleSubtitle</span>
<span class="p">}</span> <span class="n">UITableViewCellStyle</span><span class="p">;</span>
…定义适合特性参数的
1 | UITableViewCellStyle |
类型。
然而,之前苹果自己的代码中都用这种方法来定义
1 | enum |
:
1
2
3
4
5
6
7
8
<span class="k">typedef</span> <span class="k">enum</span> <span class="p">{</span>
<span class="n">UITableViewCellStyleDefault</span><span class="p">,</span>
<span class="n">UITableViewCellStyleValue1</span><span class="p">,</span>
<span class="n">UITableViewCellStyleValue2</span><span class="p">,</span>
<span class="n">UITableViewCellStyleSubtitle</span>
<span class="p">};</span>
<span class="k">typedef</span> <span class="bp">NSInteger</span> <span class="n">UITableViewCellStyle</span><span class="p">;</span>
…这种方法给出了
1 | UITableViewCellStyle |
确定的大小,但并没有告诉编译器这个类型和之前的
1 | enum |
有什么关系。
让我感动的是苹果终于给了这个“宏统一”的
1 | NS_ENUM |
。
1
NS_ENUM
1 | NS_ENUM |
从现在开始
1 | UITableViewCellStyle |
的定义已经变成这个样子了:
1
2
3
4
5
6
<span class="k">typedef</span> <span class="nf">NS_ENUM</span><span class="p">(</span><span class="bp">NSInteger</span><span class="p">,</span> <span class="n">UITableViewCellStyle</span><span class="p">)</span> <span class="p">{</span>
<span class="n">UITableViewCellStyleDefault</span><span class="p">,</span>
<span class="n">UITableViewCellStyleValue1</span><span class="p">,</span>
<span class="n">UITableViewCellStyleValue2</span><span class="p">,</span>
<span class="n">UITableViewCellStyleSubtitle</span>
<span class="p">};</span>
1 | NS_ENUM |
的第一个参数是用于存储的新类型的类型。在64位环境下,
1 | UITableViewCellStyle |
和
1 | NSInteger |
一样有8bytes长。你要保证你给出的所有值能被该类型容纳,否则就会产生错误。第二个参数是新类型的名字。大括号里面和以前一样,是你要定义的各种值。
这种实现方法提取了之前各种不同实现的优点,甚至有提示编辑器在进行
1 | switch |
判断时检查类型匹配的功能。
1
NS_OPTIONS
1 | NS_OPTIONS |
1 | enum |
也可以被定义为按位掩码(bitmask)。用简单的
1 | OR |
(
1 | | |
)和
1 | AND |
(
1 | & |
)数学运算即可实现对一个整型值的编码。每一个值不是自动被赋予从0开始依次累加1的值,而是手动被赋予一个带有一个bit偏移量的值:类似
1 | 1 << 0 |
、
1 | 1 << 1 |
、
1 | 1 << 2 |
等。如果你能够心算出每个数字的二进制表示法,例如:
1 | 10110 |
代表 22,每一位都可以被认为是一个单独的布尔值。例如在UIKit中,
1 | UIViewAutoresizing |
就是一个可以表示任何flexible top、bottom、 left 或 right margins、width、height组合的位掩码。
不像
1 | NS_ENUM |
,位掩码用
1 | NS_OPTIONS |
宏。
语法和
1 | NS_ENUM |
完全相同,但这个宏提示编译器值是如何通过位掩码
1 | | |
组合在一起的。同样的,注意值的区间不要超过所使用类型的最大容纳范围。
1 | NS_ENUM |
和
1 | NS_OPTIONS |
都是Objective-C开发中的提升开发体验的新特性,也再次展示了这门语言在对象化和过程化之间健康和谐的辩证关系。记住这一点,它就好像在你成长的道路中认识到的:我们身边的万物都是运作在矛盾且共存的严谨逻辑关系中。
转自 http://nshipster.cn/ns_enum-ns_options/