1.使用层面的理解
在这里首先讲如何简单的使用, 仅仅是使用层面(有理解错误的地方帮忙纠正), 然后我们在去理解位运算符! 在下面的代码中我们可以看见枚举值中有<<(位运算符:左移):
typedef NS_OPTIONS(NSUInteger, UIViewAutoresizing) {
    UIViewAutoresizingNone                 = 0,
    UIViewAutoresizingFlexibleLeftMargin   = 1 << 0,
    UIViewAutoresizingFlexibleWidth        = 1 << 1,
    UIViewAutoresizingFlexibleRightMargin  = 1 << 2,
    UIViewAutoresizingFlexibleTopMargin    = 1 << 3,
    UIViewAutoresizingFlexibleHeight       = 1 << 4,
    UIViewAutoresizingFlexibleBottomMargin = 1 << 5
};如果我们在枚举值中看见<<那我们就可以通过|(位运算符:或)进行组合使用如下代码为例:
 //随便添加一个UITextField
 UITextField *field = [UITextField new];
 //Begin,Changed,DidEnd都能触发UITextField的事件
 [field addTarget:self action:@selector(textFieldDidChanged) forControlEvents: 
 UIControlEventEditingDidBegin | UIControlEventValueChanged | UIControlEventEditingDidEnd];
 [self.view addSubview:field];如下图枚举值中没有<<,这就是普通的NSInteger类型的枚举, 所以不能组合使用:
typedef NS_ENUM(NSInteger, UIControlContentVerticalAlignment) {
    UIControlContentVerticalAlignmentCenter        = 0,
    UIControlContentVerticalAlignmentTop           = 1,
    UIControlContentVerticalAlignmentBottom        = 2,
    UIControlContentVerticalAlignmentFill          = 3,
};那苹果官方是怎么知道我们多个条件组合使用了呢? 答案是通过&(位运算符:与)进行判断的:
//controlEvents是组合使用后的一个值
NSUInteger controlEvents = UIControlEventEditingDidBegin | UIControlEventValueChanged | UIControlEventEditingDidEnd;
    /**    
    //通过 & 来判断是否包含:
      UIControlEventEditingDidBegin,    
      UIControlEventValueChanged,
      UIControlEventEditingDidEnd
    */
    if (controlEvents & UIControlEventEditingDidBegin) {
        NSLog(@"UIControlEventEditingDidBegin");
    }else if (controlEvents & UIControlEventValueChanged) {
        NSLog(@"UIControlEventValueChanged");
    }else if (controlEvents & UIControlEventEditingDidEnd) {
        NSLog(@"UIControlEventEditingDidEnd");
    }那么我们接下来看看使用过程中牵扯到的位运算符, 我们会在下面举个例子!
2.理解位运算符
首先我们有一个枚举, 下面代码2种写法我们暂时先不用管,等位运算符讲完我们会讨论枚举的宏使用:
//typedef NS_OPTIONS(NSInteger, myTests) {
//    nameA = 1 << 0,
//    nameB = 1 << 1,
//    nameC = 1 << 2,
//    nameD = 1 << 3,
//};
typedef enum {
    nameA = 1 << 0,
    nameB = 1 << 1,
    nameC = 1 << 2,
    nameD = 1 << 3,
}myTests;
/** 
nameA = 1 << 0 :值为1(2的0次方) 
nameB = 1 << 1 :值为2(2的1次方) 
nameC = 1 << 2 :值为4(2的2次方) 
nameD = 1 << 3 :值为8(2的3次方) 
*/通过&进行判断我们来看看输出结果如下图:
NSInteger value = nameA | nameB;
NSLog(@"%@, %@, %@, %@,",@(value & nameA),@(value & nameB),@(value & nameC),@(value & nameD));
//输出结果:1, 2, 0, 0,我们得到NSInteger value = nameA | nameB;的组合的值, 判断结果是:1是nameA的值, 2是nameB的值, nameC与nameD没有组合使用所以值为0,最后我们知道如果value & nameC为0说明value不包含nameC 相反则包含!
还有一点就是value & nameA就是nameA的值为1, value & nameB就是nameB的值为2
- <<(左移):- a << b就表示把- a转为二进制后左移- b位(在后面添- b个- 0)
- |(或):只要有一个为- 1, 结果就是- 1
- &(与):只要有二个为- 1, 结果才是- 1
我们已经知道nameA = 1, nameB = 2, nameC = 4, nameD = 8下面来通过二进制来解释:
 NSInteger value = nameA | nameB | nameC | nameD;
     转成二进制:
     nameA: 0 0 0 1
       |
     nameB: 0 0 1 0
       |
     nameC: 0 1 0 0
       |
     nameD: 1 0 0 0
    ----------------
     value: 1 1 1 1
     上面是使用 | 得出value的值为1111(|的意思是有一个为1结果就为1)
     下面是使用 & 判断输出的值(&的意思就是有二个为1结果才为1)
      value: 1 1 1 1         value: 1 1 1 1
        &                      &
      nameA: 0 0 0 1         nameB: 0 0 1 0
     ----------------       ----------------
      结果值: 0 0 0 1         结果值: 0 0 1 0
      我就写2个例子:0001就是nameA的值, 0010就是nameB的值3.枚举的宏(NS_ENUM与NS_OPTIONS)
NS_ENUM和NS_OPTIONS宏提供了一个简洁、定义枚举和C语言选项的简单方法。
The NS_ENUM and NS_OPTIONS macros provide a concise, simple way of defining enumerations and options in C-based languages. These macros improve code completion in Xcode and explicitly specify the type and size of your enumerations and options. Additionally, this syntax declares enums in a way that is evaluated correctly by older compilers, and by newer ones that can interpret the underlying type information.
这是最初的使用方法:
enum {
        UITableViewCellStyleDefault,
        UITableViewCellStyleValue1,
        UITableViewCellStyleValue2,
        UITableViewCellStyleSubtitle
};
typedef NSInteger UITableViewCellStyle;
--------------------------------------------------
enum {
        UIViewAutoresizingNone                 = 0,
        UIViewAutoresizingFlexibleLeftMargin   = 1 << 0,
        UIViewAutoresizingFlexibleWidth        = 1 << 1,
        UIViewAutoresizingFlexibleRightMargin  = 1 << 2,
        UIViewAutoresizingFlexibleTopMargin    = 1 << 3,
        UIViewAutoresizingFlexibleHeight       = 1 << 4,
        UIViewAutoresizingFlexibleBottomMargin = 1 << 5
};
typedef NSUInteger UIViewAutoresizing;通过使用枚举的宏:
NS_ENUM:是用来声明一般的NSInteger(下面代码使用NSInteger)类型的枚举
Use the NS_ENUM macro to define enumerations, a set of values that are mutually exclusive.
NS_OPTIONS:是用来声明位掩码(bitmasked)
Use the NS_OPTIONS macro to define options, a set of bitmasked values that may be combined together.
//NS_ENUM
typedef NS_ENUM(NSInteger, UITableViewCellStyle) {
        UITableViewCellStyleDefault,
        UITableViewCellStyleValue1,
        UITableViewCellStyleValue2,
        UITableViewCellStyleSubtitle
};
--------------------------------------------------
//NS_OPTIONS
typedef NS_OPTIONS(NSUInteger, UIViewAutoresizing) {
        UIViewAutoresizingNone                 = 0,
        UIViewAutoresizingFlexibleLeftMargin   = 1 << 0,
        UIViewAutoresizingFlexibleWidth        = 1 << 1,
        UIViewAutoresizingFlexibleRightMargin  = 1 << 2,
        UIViewAutoresizingFlexibleTopMargin    = 1 << 3,
        UIViewAutoresizingFlexibleHeight       = 1 << 4,
        UIViewAutoresizingFlexibleBottomMargin = 1 << 5
};NS_OPTIONS 与 NS_ENUM 和 enum 是有什么区别呢?
1.通过上面介绍我们可以看出enum可以声明一般类型和位掩码(bitmasked)类型
2.NS_ENUM声明一般类型, NS_OPTIONS声明掩码(bitmasked)类型
3.那么问题又来了, 直接用enum不就可以了? 答案不是这样的, 苹果建议我们在OC中使用NS_ENUM与NS_OPTIONS, 为什么呢? 因为他们除了推断出不同类型的枚举,再就是当编译Objective-C++模式,它们产生的代码是不同的, 就是因为不同所以混编的时候使用enum会报错!
参考:
iOS - What is the difference between NS_ENUM and NS_OPTIONS? - Stack Overflow