0、背景
在iOS6和Mac OS 10.8以后Apple引入了两个宏来重新定义这两个枚举类型,实际上是将enum定义和typedef合二为一,并且采用不同的宏来从代码角度来区分。
1、目的:
增加代码的可读性
2、本质:
是一个整形(int)
并且,它不参与内存的占用和释放,枚举定义变量即可直接使用,不用初始化;
枚举值一般是4个字节的int值,在64位系统上是8个字节。
3、两种使用:NS_ENUM、NS_OPTIONS
3.0 enum:C的写法
//C的写法
typedef enum {
//以下是枚举成员
EnumNameRight = 0,
EnumNameLeft = 1,
EnumNameUp = 2,
EnumNameDown = 3
} EnumName;
3.1 NS_ENUM:是用来声明一般的NSInteger类型,不可多选组合使用;
枚举成员也可不用每一个都赋值,第一个赋值就行了,后面的以此加一类推。
typedef NS_ENUM(NSInteger, EnumName) {
//以下是枚举成员
EnumNameRight = 0,
EnumNameLeft = 1,
EnumNameUp = 2,
EnumNameDown = 3
};
3.2 NS_OPTIONS:是用来声明位掩码(bitmasked)类型,有左位移操作或特点的情况,可以多选组合使用
typedef NS_OPTIONS(NSInteger, EnumName) {
//以下是枚举成员
EnumNameNone = 0, // 对应的十进制0
EnumNameRight = 1 << 0, // 对应的十进制1(2的0次方) 对应二进制 0001(1不移动)
EnumNameLeft = 1 << 1, // 对应的十进制2(2的1次方) 对应二进制 0010(1往左移1位)
EnumNameUp = 1 << 2, // 对应的十进制4(2的2次方) 对应二进制 0100(1往左移2位)
EnumNameDown = 1 << 3 // 对应的十进制8(2的3次方) 对应二进制 1000(1往左移3位)
};
3.3 定义枚举属性
//定义一个枚举属性,代表该枚举值
@property (nonatomic, assign) EnumName enumName;
4、或运算 |
NSInteger enumNames = EnumNameRight | EnumNameLeft | EnumNameDown;
其运算结果是在二进制对应的位置设置把0设置成1。
所以有位移操作特点才能使用或运算,否则会得出不是我们想要的答案。
比如:NSInteger enumCount = 1 | 3 ; 结果是等于3;
相当于NSInteger enumCoun = (0001) | (0011) ;
5、与运算 &
苹果官方是怎么知道我们多个条件组合使用了呢? 答案是通过&(位运算符:与)进行判断的:
//controlEvents是组合使用后得出的一个值
NSUInteger controlEvents = UIControlEventEditingDidBegin | UIControlEventValueChanged
/**
//通过 & 来判断是否包含:
UIControlEventEditingDidBegin,
UIControlEventValueChanged,
*/
if (controlEvents & UIControlEventEditingDidBegin) {
NSLog(@"UIControlEventEditingDidBegin");
}else if (controlEvents & UIControlEventValueChanged) {
NSLog(@"UIControlEventValueChanged");
}
6、位运算总结
<<(左移):a << b就表示把a转为二进制后左移b位(在后面添b个0)
|(或):只要有一个为1, 结果就是1
&(与):只要有二个为1, 结果才是1
7、对位运算结果的加和减操作
NSInteger value = EnumNameLeft | EnumNameDown;(1 | 3)
//累加
value |= EnumNameDown; //结果为7
//累减
value ^= EnumNameDown; //结果为0
8、NS_OPTIONS
与 NS_ENUM
和 enum
是有什么区别
1.通过上面介绍我们可以看出enum可以声明一般类型
和位掩码(bitmasked)类型
2.NS_ENUM
声明一般类型, NS_OPTIONS
声明掩码(bitmasked)类型
3.那么问题又来了, 直接用enum
不就可以了? 答案不是这样的, 苹果建议我们在OC中使用NS_ENUM
与NS_OPTIONS
, 为什么呢? 因为他们除了推断出不同类型的枚举,再就是当编译Objective-C++模式,它们产生的代码是不同的, 就是因为不同所以混编的时候使用enum
会报错!