查看原文
其他

弄懂Android 源码中那些巧妙位运算

猛猛的小盆友 鸿洋 2019-04-05

本文作者


作者:猛猛的小盆友

链接:

https://www.jianshu.com/p/5f41b3cc1909

本文由作者授权发布。


1前言


在查看源码中,经常会看到很多这样的符号“&”、“|”、“~”,咋一看挺高大上;仔细一看,有点懵;再看看,其实就是大学学过的再普通不过的与、或、非。今天小盆友就以简单的形式分享下,同时也是作为笔记记录,等某天突然懵逼时可以回来看看,话不多说,开始撸。


2与或非是啥?


或许,大学上的课,在还没毕业时就还给老师了,所以我们先重温下。


1、与运算符 &


知识点:两位同时为“1”,结果才为“1”,否则为“0”。


运算规则:0&0=0;  0&1=0;   1&0=0;    1&1=1;


System.out.println("0 & 0" + (0 & 0));
System.out.println("0 & 1" + (0 & 1));
System.out.println("1 & 0" + (1 & 0));
System.out.println("1 & 1" + (1 & 1));



其实就是运算的位要完全一样,才保持原样,否则就变为0。


2、或运算符 |


知识点:只要有一位为1,其值为1,否则位0。


运算规则:0|0=0;  0|1=1;  1|0=1;   1|1=1;


System.out.println("0 | 0 = " + (0 | 0));
System.out.println("0 | 1 = " + (0 | 1));
System.out.println("1 | 0 = " + (1 | 0));
System.out.println("1 | 1 = " + (1 | 1));



其实就是只要有1,结果就为1。


3、非运算符 ~


知识点:如果位为0,结果是1。如果位为1,结果是0

运算规则:~0=1;  ~1=0;


System.out.println("~1 = " + ~(1));
System.out.println("~0 = " + ~(0));



很惊喜!很意外!有没有?!竟然不是0和1,这里不是计算机出问题了,而是涉及到了计算机内部的编码的问题,是不是想到了大学有一门课叫做《计算机科学导论》。


详细讲解可以查看这里=>为了方便查看,我从文章截了一张图

https://bbs.csdn.net/wap/topics/100095663



3实战


逼逼叨了这么多,其实位运算符的文章很多,这里其实还少了一个异或运算符,但因为没有出现在实战中,所以就不做多余的操作了。我们来进行真正的运用吧。


1、场景一(或运算符的使用)


你有没有在xml中这样编写过布局


android:layout_gravity="bottom|right"


我们这里就不用bottom、right在源码中真实的值,以方便讲解


这里的 bottom 和 right 在位上肯定是错开的,这样做位运算时,才能同时保存该控件 “居右”和“底部” 的属性。


什么叫位上错开,且看下面代码。


// 0x001 = 0000 0001 
int right = 0x001;
// 0x001 = 0000 0010 
int bottom = 0x002;
// 结果 = 0000 0011 = 3
System.out.println("right | bottom = " + (right | bottom));



通过上面的代码,或许你已经恍然大悟(雾?),其实位错开是为了或运算时,进行值的保留。 


让两个状态的能够保存在一个属性中,或许你会问这样有什么好处了?我挠了下头,想到了以下三个好处:


  1. 节省空间,避免不必要的属性出现和维护成本(难道你想一个状态用一个布尔值来维护么?)

  2. 获取方便,编码简洁,位运算也更加高效

  3. 装bi,不装bi的程序员不是好的搬砖工(~)


2、场景二(与运算符的使用)


上一小节说的是如何组装成一个值,要怎么使用它呢?


安卓源码中怎么知道我们设置了 right 这个居右的状态呢?这个便需要使用 “与” 运算符来 取值。


具体操作如下代码:


int right = 0x001;
int bottom = 0x002;
int top = 0x008;
int state = right | bottom;
System.out.println("是否存在 right = " + ((state & right) == right));
System.out.println("是否存在 top = " + ((state & top) == top));



从上面的代码很清晰的看出,用 “与” 运算符进行 “取值”。是不是有点小惊喜呢?


3、场景三(非运算符的使用)


或许,你会有这样的一个疑问,如果我想剔除当前已经包含的一个值,需要怎么办?这时候就是“非”和“与”运算符联合使用的时候了,且看下面代码


int right = 0x001;
int bottom = 0x002;
int top = 0x008;
int state = right | bottom;
System.out.println("剔除 right 状态前 " + state);
state &= ~right;
System.out.println("剔除 right 状态后 " + state);
state &= ~top;
System.out.println("剔除不存在的 top 状态 " + state);
System.out.println();




是不是有点小激动了呢?哈哈,在安卓源码中运用挺多,举个栗子🌰,在ViewGroup中的requestDisallowInterceptTouchEvent方法便有用到,这里就不一一列举。


敲黑板啦!!! 为什么上面能做到剔除呢?小盆友手写了下过程(多年没写字,不要喷,哈哈哈哈)



四、小结


  • 或运算符整合值

  • 与运算符取值

  • 与非剔除值


五、写在最后


或许还有很多更好玩的用法,但限于小盆友能力有限,文笔也一般般,所以就只能到这啦。如果您有更好或是更有趣的用法,或是本文有不妥之处,请与分享和纠正。编码使我快乐,哈哈哈。

                   

推荐阅读

推荐2个走心项目

Toast 不显示了?



扫一扫 关注我的公众号

如果你想要跟大家分享你的文章,欢迎投稿~


┏(^0^)┛明天见!

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存