枚舉增強程式的可讀性,用法上還是需要註意的 1.C語言的寫法 enum XMPPReconnectFlags { kShouldReconnect = 1 << 0, // If set, disconnection was accidental, and autoReconnect may be ...
枚舉增強程式的可讀性,用法上還是需要註意的
1.C語言的寫法
enum XMPPReconnectFlags
{
kShouldReconnect = 1 << 0, // If set, disconnection was accidental, and autoReconnect may be used
kMultipleChanges = 1 << 1, // If set, there have been reachability changes during a connection attempt
kManuallyStarted = 1 << 2, // If set, we were started manually via manualStart method
kQueryingDelegates = 1 << 3, // If set, we are awaiting response(s) from the delegate(s)
};
或者是
typedef enum XMPPReconnectFlags
{
kShouldReconnect = 1 << 0, // If set, disconnection was accidental, and autoReconnect may be used
kMultipleChanges = 1 << 1, // If set, there have been reachability changes during a connection attempt
kManuallyStarted = 1 << 2, // If set, we were started manually via manualStart method
kQueryingDelegates = 1 << 3, // If set, we are awaiting response(s) from the delegate(s)
}XMPPReconnectFlags;//此處的XMPPReconnectFlags為別名
2.OC的寫法
typedef NS_ENUM(NSInteger, XMPPMessageBodyType) {
XMPPMessageBodyTypeText = 1, //文本
XMPPMessageBodyTypeImage, //圖片
XMPPMessageBodyTypeSound, //語音
XMPPMessageBodyTypeMap, //地理信息(文本)
XMPPMessageBodyTypeNotification, //通知
XMPPMessageBodyTypeCustom, //自定義
XMPPMessageBodyTypeVideo, //視頻
XMPPMessageBodyTypeExpression, //表情
};
第一個枚舉值只要NSInteger類型,後面的依次累加,中間再重新賦值,後面的會從賦值處重新累加
3.NS_ENUM和NS_OPTIONS
NS_ENUM和NS_OPTIONS本質是一樣的,僅僅從字面上來區分其用途。NS_ENUM是通用情況,NS_OPTIONS一般用來定義具有位移操作或特點的情況
typedef NS_OPTIONS(NSInteger, ShareViewItemType) {
ShareViewItemTypeWX = 1,
ShareViewItemTypeWXFC = 1 << 0,
ShareViewItemTypeQQ= 1 << 1,
ShareViewItemTypeWB= 1 << 1,
ShareViewItemTypeReport= 1 << 3,
ShareViewItemTypeBarrage = 1 << 4,
ShareViewItemTypeAll = 1 << 5,
};
如果多選枚舉,那麼需要這樣判斷
+ (instancetype)shareViewWithDic:(NSDictionary*)dic shareViewItemType:(ShareViewItemType)shareViewItemType;
{
if (shareViewItemType == (ShareViewItemTypeWX|ShareViewItemTypeWXFC)) {
// code
}
}
4.官方的全選枚舉
UIControl
的枚舉
typedef NS_OPTIONS(NSUInteger, UIControlEvents) {
UIControlEventTouchDown = 1 << 0, // on all touch downs
UIControlEventTouchDownRepeat = 1 << 1, // on multiple touchdowns (tap count > 1)
UIControlEventTouchDragInside = 1 << 2,
UIControlEventTouchDragOutside = 1 << 3,
UIControlEventTouchDragEnter = 1 << 4,
UIControlEventTouchDragExit = 1 << 5,
UIControlEventTouchUpInside = 1 << 6,
UIControlEventTouchUpOutside = 1 << 7,
UIControlEventTouchCancel = 1 << 8,
UIControlEventValueChanged = 1 << 12, // sliders, etc.
UIControlEventPrimaryActionTriggered NS_ENUM_AVAILABLE_IOS(9_0) = 1 << 13, // semantic action: for buttons, etc.
UIControlEventEditingDidBegin = 1 << 16, // UITextField
UIControlEventEditingChanged = 1 << 17,
UIControlEventEditingDidEnd = 1 << 18,
UIControlEventEditingDidEndOnExit = 1 << 19, // 'return key' ending editing
UIControlEventAllTouchEvents = 0x00000FFF, // for touch events
UIControlEventAllEditingEvents = 0x000F0000, // for UITextField
UIControlEventApplicationReserved = 0x0F000000, // range available for application use
UIControlEventSystemReserved = 0xF0000000, // range reserved for internal framework use
UIControlEventAllEvents = 0xFFFFFFFF
};
最後一個 UIControlEventAllEvents
賦值為0xFFFFFFFF,意為全選,實際工程使用沒有問題,很方便,但是在
Archive的時候會報錯
屏幕快照 2016-06-29 下午12.28.10.png
這個地方需要這樣改
typedef NS_OPTIONS(NSUInteger, ShareViewItemType) {
ShareViewItemTypeWX = 1,
ShareViewItemTypeWXFC = 1 << 0,
ShareViewItemTypeQQ= 1 << 1,
ShareViewItemTypeWB= 1 << 1,
ShareViewItemTypeReport= 1 << 3,
ShareViewItemTypeBarrage = 1 << 4,
ShareViewItemTypeAll = 0xFFFFFFFF,
};
你可以這樣判斷條件
+ (instancetype)shareViewWithDic:(NSDictionary*)dic shareViewItemType:(ShareViewItemType)shareViewItemType;
{
// 入參shareViewItemType為ShareViewItemTypeAll
if (shareViewItemType & ShareViewItemTypeBarrage) {
// code
}
}
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
類型的枚舉, 所以不能組合使用:
那蘋果官方是怎麼知道我們多個條件組合使用了呢? 答案是通過&
(位運算符:與)進行判斷的:
//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");
}
那麼我們接下來看看使用過程中牽扯到的位運算符, 我們會在下麵舉個例子!
6.理解位運算符
首先我們有一個枚舉, 下麵代碼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;
的組合的值, 判斷結果是: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的值
相信大家已經明白其中的道理了, 接下來我們來看看枚舉的巨集, 為了更好閱讀也可以看下麵的截圖:
7.枚舉的巨集(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
會報錯!