iOS 开发iOS开发iOS学习

iOS(三)OC中的枚举(NS_ENUM和NS_OPTION)

2017-09-22  本文已影响178人  ePan

对于OC中的枚举类型,虽然知道有NS_ENUM和NS_OPTION,然而并不是十分清楚它们之间的区别。另外,也很好奇,OC中为什么几乎不使用enum,以及这三种枚举类型之间的差异。

通过查阅资料,同时编写代码调试,总算弄清楚了之前的问题,所以写这篇文章来进行总结。

enum的局限性

在枚举的声明中,使用typedef可以使得枚举变量的声明更简单。

typedef enum {
    FlyStateOne,
    FlyStateTwo,
    FlyStateThree
}FlyState;

enum FlyTypeState{
    FlyTypeOne,
    FlyTypeTwo,
    FlyTypeThree
};
FlyState  state;
typedef enum  FlyTypeState state;

C++11标准扩充了枚举的类型,通过新式枚举,可以指明枚举的底层数据类型。这样,就可以确定给枚举变量分配多少空间,从而能够向前声明枚举变量。

而对于enum来说,不能直接在使用typedef的同时,确定枚举的底层数据类型。

typedef enum {
    FlyStateOne,
    FlyStateTwo,
    FlyStateThree
}FlyState;

enum FlyState:NSInteger{ //设置底层数据类型为NSInteger
    FlyStateOne,
    FlyStateTwo,
    FlyStateThree
};

上面的两种声明方式,都是正确的,而下面的这种声明方式是错的:

typedef enum FlyState1:NSInteger{
    FlyStateOne1,
    FlyStateTwo1,
    FlyStateThree1
};

enum如果想在使用typedef的同时,确定底层数据类型,就只能使用两条语句:

typedef enum FlyState:NSInteger FlyState;
enum FlyState : NSInteger {
    FlyStateOne,
    FlyStateTwo,
    FlyStateThree
};

这样,很显然在写代码的时候比较麻烦。

NS_ENUM和enum的不同

NS_ENUM是一个OC中的宏,可以判断编译器能否采用新式枚举:如果不能,那么效果等同于仅仅使用typedef的enum;如果能够采用新式枚举,那么NS_ENUM所定义的枚举类型,就是处理后的enum类型,可以在使用typedef的同时,指定底层数据类型。
比如:

typedef NS_ENUM(NSInteger, FlyState) {
    FlyStateOne,
    FlyStateTwo,
    FlyStateThree
};

如果支持新式枚举,展开之后,就是:

typedef enum FlyState:NSInteger FlyState;
enum FlyState : NSInteger {
    FlyStateOne,
    FlyStateTwo,
    FlyStateThree
};

此时,既指定了数据类型,又可以使用:

FlyState state = FlyStateOne; 

很明显,NS_ENUM就是对enum的一种封装。所以,在OC中,几乎不使用enum,都是使用NS_ENUM和NS_OPTION。

按位或 枚举的作用

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
};

需要同时满足多个枚举条件时,使用按位或操作来进行组合。 比如:

UIViewAutoresizingFlexibleRightMargin|UIViewAutoresizingFlexibleTopMargin

用于表示同时限制右边距和头部边距

NS_ENUM和NS_OPTIONS的不同

NS_OPTIONS也是OC中的一个宏。

在用或运算处理两个枚举类型时,C++ 认为枚举结果的数据类型应该是枚举的底层数据类型(即NSUInteger),而且C++不允许将这个底层类型隐式转换成枚举类型本身。
比如:

typedef NS_ENUM(NSUInteger, UIViewAutoresizing) {
    UIViewAutoresizingNone                 = 0,
    UIViewAutoresizingFlexibleLeftMargin   = 1 << 0,
    UIViewAutoresizingFlexibleWidth        = 1 << 1,
    UIViewAutoresizingFlexibleRightMargin  = 1 << 2,
    UIViewAutoresizingFlexibleTopMargin    = 1 << 3,
    UIViewAutoresizingFlexibleHeight       = 1 << 4,
    UIViewAutoresizingFlexibleBottomMargin = 1 << 5
};

此时:

UIViewAutoresizing viewModel =UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleHeight

得到的枚举结果的类型是NSUInteger,而不是UIViewAutoresizing。编译器会报错。

针对这种情况,NS_OPTIONS做了处理,区分了C++ 模式编译和非C++ 模式编译。在非C++ 编译模式下,NS_OPTIONS和NS_ENUM是一样的。对于C++模式编译,NS_OPTIONS会做一些特殊处理,保证枚举结果的类型正确。

总结

NS_ENUM的作用是在使用typedef的同时,确定枚举的底层数据类型,这个效果是enum所达不到的。NS_OPTIONS的作用是,在NS_ENUM的基础上,使得按位或运算的结果能够返回正确的数据类型。

所以,凡是需要以按位或操作来进行组合的枚举都需要用NS_OPTIONS定义,若不需要组合,则可以使用NS_ENUM。


文中如有错误,欢迎指正。

上一篇 下一篇

猜你喜欢

热点阅读