十六进制常量还有这种玩法

412 阅读5分钟

持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第9天,点击查看活动详情

前言

上一篇文章中 juejin.cn/post/715437… ,在源码解析阶段,那些判断16进制的地方,很有意思,加上我以前也写过一篇关于这个的文章 www.jianshu.com/p/bff2b84ca… ,所以想在这里做个分享。

状态变量的一般写法

可能有些朋友平时在开发时定义常量状态是这样定义的:

public static final int SEX_BOY = 0; // 男生
public static final int SEX_GIRL = 1; // 女生

然后看了某篇文章之后,某个经验丰富的说,定义常量时最好使用16进制,再去看了看Android某些类的源码,嗯,好像里面定义常量确实是使用了16进制,于是之后写代码就开始

public static final int SEX_BOY = 0x00; // 男生
public static final int SEX_GIRL = 0x01; // 女生
public static final int SEX_OTHER = 0x02; // 其它

比如说有很多个状态,就从0x01、0x02......0xff 这样列举下去。

这样写对吗?我随便找个源码来举例下,随便从View.java扣出一段代码

e60a26626531e0f7231f1858adc1518.png

为什么是这样定义呢,为什么不是像我们那种写法?

叠加状态的定义方式

其实这个直接说不好解释,跟着我去操作,就理解为什么要这么定义了。

假设我们定义状态,定义成这样

public static final int TYPE1 = 0x01;
public static final int TYPE2 = 0x02;
public static final int TYPE3 = 0x04;
public static final int TYPE4 = 0x08;
public static final int TYPE5 = 0x10;
public static final int TYPE6 = 0x20;
public static final int TYPE7 = 0x40;
public static final int TYPE8 = 0x80;

为什么这么写呢?
我们将16进制转成2进行,上面就对应成

0d39ccd1271903a3cc6f461d2b03217.png

有意思的就在这里,我先说的我想的过程中错误的一个思路 (我觉得挺有意思的,所以可以说下,因为是一个错误的思路,如果不想看可以直接跳看下面的这样定义的原因)

二进制从右往左来说
(1)我用第一位表示性别 000:女 001:男
(2)我用第二位表示角色 000:学生 010:老师
(3)我用第三位表示班级 000:A班 100:B班
那么 “A班的女老师” 我可以表示成 010 = 2
“A班的男老师” 可以表示成 011=3
“B班的女学生”可以表示成 100 .....
这样可以组成8个状态而不会冲突,但是这样的做法是只能用3个状态组合进行比较,而且单个状态下有000表示了3种,而且这种做法同一位上只能表示两种状态,假如我加个C班,那就没辙了。

然后换了一种思考的方法,假如我这样表示状态

public static final int TYPE1 = 0x01;  // 女
public static final int TYPE2 = 0x02;  // 男
public static final int TYPE3 = 0x04;  // 学生
public static final int TYPE4 = 0x08;  // 老师
public static final int TYPE5 = 0x10;  // 主任
public static final int TYPE6 = 0x20;  // A班
public static final int TYPE7 = 0x40;  // B班
public static final int TYPE8 = 0x80;  // C班

那么 使用二进制的或运算:
“A班的女老师” 我可以表示成 TYPE6|TYPE1|TYPE4 = 0010 1001 = 41
“A班的男老师” 可以表示成 TYPE6|TYPE2|TYPE4 = 0010 1010 = 42
“B班的女主任”可以表示成 TYPE7|TYPE1|TYPE5 = 0101 0001 = 81
这样也能把多个状态组成一个状态,而且组合状态也能和单个状态进行同等级判断,并且这种做法不会产生重复的状态。

举个例子就是说你平时写

if(性别==女 && 角色 == 老师 && 班级 == A班){
    ......
}else if(版本 == C班){
    ......
}

如果用我这种方法定义状态的话,你只用写

if(type == 0x29){
    ......
}else if(type == 0x80){
    ......
}

可能有些人说就仅仅为了这样?那我写&&还好过,写成16进制转换转的我脑壳疼。我还不如多写几个&&,而且这样也更容易让别人看懂。 但这个写法不仅仅有这种好处,再举个例子,假如在很多个组合的状态中你需要去判断这个状态是“男”还是“女”等等,多状态下判断单状态多了,也不是说乱,但会写很多代码,但是现在可以直接这样写

public void switchSex(type){
    if(type & 0x03 == 0x01){
        // 是女生
    }else{
        // 是男生
    }
}

就可以直接这样用二进制的与运算来实现判断。
我也仅仅是举了两个例子,我的意思是这样去定义十六进制常量,方便二进制做运算,二进制还有其他的运算呢,我仅仅举了“或”和“与”,还有什么异或啊,移位啊之类的,而且就算作用不大,按装逼来说,我直接做二进制的运算肯定比你那些乱七八糟的运算来得快吧。

总结

当然这只是我领悟的一种思路,而且我想很多人也知道这种做法,或者用16进制来定义常量不仅仅有这个好处,只是我觉得很有意思,所以想分享一下。