美文网首页
iOS开发-之-位移枚举和按位或(|)按位与(&)运算

iOS开发-之-位移枚举和按位或(|)按位与(&)运算

作者: eryuxinling | 来源:发表于2016-12-09 10:54 被阅读191次

    前言

    Enumeration,enum,在程序设计语言中,一般用一个数值来代表某一状态,这种处理方法不直观,易读性差。如果能在程序中用自然语言中有相应含义的单词来代表某一状态,则程序就很容易阅读和理解。也就是说,事先考虑到某一变量可能取的值,尽量用自然语言中含义清楚的单词来表示它的每一个值,这种方法称为枚举方法,用这种方法定义的类型称枚举类型。

    命名

    定义的枚举类型名称通常与项目的类文件前缀相同,或者是类库框架缩写,或者跟随具体业务名,如果开头是缩写要大写表示,跟随其后的命名应采用驼峰命名法则,命名应准确表述枚举表示的意义,枚举中各个值都应以定义的枚举类型开头,其后跟随各个枚举值对应的状态、选项或者状态码

    状态与选项(states and options)

    状态
    同时只能有一种,如“RecordPause”,“RecordEnd”,不可能同时Record是RecordPause和RecordEnd。如下:

    typedef NS_ENUM(NSUInteger, VoiceRecordStatus){
        VoiceRecordStatusPrepare = 0,
        VoiceRecordStatusIng,
        VoiceRecordStatusPause,
        VoiceRecordStatusEnd
    };
    

    由于每种状态都用一个便于理解的值来表示,所以这样写出来的代码更易读懂。编译器会为枚举分配一个独有的编号,从0开始,每个枚举递增1

    选项

    定义选项的时候。若这些选项可以彼此组合,则更应如此。只要枚举定义得对,各选项之间就可通过“按位或操作符”(bitwise OR operator)来组合。例如,iOS UI框架中有如下枚举类型,用来表示某个视图应该如何在水平或垂直方向上调整大小

    位移枚举(可复选的枚举) 使用位移实现选项变量

    本文使用 Autoresizing 的系统枚举UIViewAutoresizing作为基础进行讲述:

    以下位移已经提前计算出来了二进制与十进制值为了方便下文使用

    typedef NS_OPTIONS(NSUInteger, UIViewAutoresizing){    二进制值    十进制
        UIViewAutoresizingNone                  = 0,       00000000    0
        UIViewAutoresizingFlexibleLeftMargin    = 1<<0,    00000001    1
        UIViewAutoresizingFlexibleWidth         = 1<<1,    00000010    2
        UIViewAutoresizingFlexibleRightMargin   = 1<<2,    00000100    4
        UIViewAutoresizingFlexibleTopMargin     = 1<<3,    00001000    8
        UIViewAutoresizingFlexibleHeight        = 1<<4,    00010000    16
        UIViewAutoresizingFlexibleBottomMargin  = 1<<5     00100000    32
    };
    

    使用枚举定义选项,每个选项均可启用或禁用,使用上述方式来定义枚举值,每个枚举值所对应的二进制表示中,只有1个二进制位的值是1。用“按位或操作符”可组合多个选项

    用( | )来隔开

    aView.autoresizingMask = UIViewAutoresizingFlexibleTopMargin | UIViewAutoresizingFlexibleLeftMargin | UIViewAutoresizingFlexibleBottomMargin | UIViewAutoresizingFlexibleRightMargin;
    

    而且每一个枚举对应的逻辑都会覆盖到

    实例

    - (void)todo:(UIViewAutoresizing)type {
        if (type == 0){
            NSLog(@"UIViewAutoresizingNone");
            return;
        }
        if (type & UIViewAutoresizingFlexibleLeftMargin){
            NSLog(@"UIViewAutoresizingFlexibleLeftMargin");
        }
        if (type & UIViewAutoresizingFlexibleWidth){
            NSLog(@"UIViewAutoresizingFlexibleWidth");
        }
        if (type & UIViewAutoresizingFlexibleRightMargin){
            NSLog(@"UIViewAutoresizingFlexibleRightMargin");
        }
        if (type & UIViewAutoresizingFlexibleTopMargin){
            NSLog(@"UIViewAutoresizingFlexibleTopMargin");
        }
        if (type & UIViewAutoresizingFlexibleHeight){
            NSLog(@"UIViewAutoresizingFlexibleHeight");
        }
        if (type & UIViewAutoresizingFlexibleBottomMargin){
            NSLog(@"UIViewAutoresizingFlexibleBottomMargin");
        }
    }
    
    - (void)viewDidLoad {
        [superviewDidLoad];
        //Do any additional setup after loading the view, typically from anib.
        [selftodo:UIViewAutoresizingFlexibleLeftMargin|UIViewAutoresizingFlexibleRightMargin|UIViewAutoresizingFlexibleHeight];
    }
    结果输出
    UIViewAutoresizingFlexibleLeftMarginUIViewAutoresizingFlexibleRightMarginUIViewAutoresizingFlexibleHeight
    

    二进制转十进制

    1101(2进制)= 1x20+0x21+1x22+1x23=1+0+4+8=13转化成十进制要从右到左用二进制的每个数去乘以2的相应次方不过次方要从0开始

    位移位运算

    如 UIViewAutoresizingFlexibleHeight = 1 << 4,

    1.左移运算 1 << 4
    将一个运算对象的各二进制位全部左移若干位(左边的二进制位丢弃,右边补0)。
    1 转化为二进制为 :0000 0001
    左移四位就为 :0001 0000
    0001 0000 转化为十进制等于16

    2.右移运算 90>>4
    将一个数的各二进制位全部右移若干位,正数左补0,负数左补1,右边丢弃。操作数每右移一位,相当于该数除以2。
    90转化为二进制为 :01011010
    右移4位就是 :00000101
    00000101 转化为十进制等于5

    按位或运算符(|)

    运算规则:0|0=0; 0|1=1; 1|0=1; 1|1=1;
    即 :参加运算的两个二进制对应数位只要有一个为1,其值为1

    例如:
    [self todo:UIViewAutoresizingFlexibleLeftMargin|UIViewAutoresizingFlexibleRightMargin|UIViewAutoresizingFlexibleHeight];
    

    1.十进制1|4|16 转为二进制0000 0001 | 0000 0100 | 0001 0000 = 0001 0101,因此,1|4|16的十进制值得21

    按位与运算符(&)

    参加运算的两个数据,按二进制位进行“与”运算。
    运算规则:0&0=0; 0&1=0; 1&0=0; 1&1=1;
    即:两对应数位同时为“1”,结果才为“1”,否则为0

    1.十进制3&5 转为二进制 0000 0011 & 0000 0101 = 0000 0001 因此,3&5的值得1

    2.在iOS方法中的应用

    - (void)todo:(UIViewAutoresizing)type {
       if (type & UIViewAutoresizingFlexibleLeftMargin) {
          NSLog(@"UIViewAutoresizingFlexibleLeftMargin");
       }
    }
    传入参数type 为 UIViewAutoresizingFlexibleLeftMargin|UIViewAutoresizingFlexibleRightMargin|UIViewAutoresizingFlexibleHeight
    
    上文知道结果是 0001 0101,十进制值得21
    UIViewAutoresizingFlexibleRightMargin为4
    
    0001 0101 & UIViewAutoresizingFlexibleRightMargin = 0001 0101 & 0000 0100 =21 & 4 = 4 
    根据计算结果还是UIViewAutoresizingFlexibleRightMargin这个枚举
    所以理论上更严谨的写法应该是:
    if ((type & UIViewAutoresizingFlexibleRightMargin) == UIViewAutoresizingFlexibleRightMargin){
    }
    

    要点 引用自effective Objective-C 2.0

    应该用枚举来表示状态机的状态、传递给方法的选项以及状态码等值,给这些值起个易懂的名字。

    如果把传递给某个方法的选项表示为枚举类型,而多个选项又可同时使用,那么就将各选项值定义为2的幂,以便通过按位或操作将其组合起来。

    用NS_ENUM与NS_OPTIONS宏来定义枚举类型,并指明其底层数据类型。这样做可以确保枚举是用开发者所选的底层数据类型实现出来的,而不会采用编译器所选的类型。

    在处理枚举类型的switch语句中不要实现default分支。这样的话,加入新枚举之后,编译器就会提示开发者:switch语句并未处理所有枚举。

    相关文章

      网友评论

          本文标题:iOS开发-之-位移枚举和按位或(|)按位与(&)运算

          本文链接:https://www.haomeiwen.com/subject/musemttx.html