Android中巧妙的位运算
按计划每周更新一篇技术博文,第六篇:《Android中巧妙的位运算》
Android在编码的时候经常使用到位运算,这里以Intent的Flags为例。(查看Intent说明文档)
首先通过查看Flags的值,都是16进制数值代表,且只使用一位并只为1|2|4|8 (与2的次方相关),例举几个源码中对应的值: public static final int FLAG_ACTIVITY_NEW_TASK = 0x10000000; public static final int FLAG_ACTIVITY_SINGLE_TOP = 0x20000000; public static final int FLAG_ACTIVITY_MULTIPLE_TASK = 0x08000000;
再来看看1|2|4|8分别对应的二进制数:1 : 00012 : 00104 : 01008 : 1000
注意:它们通过“或运算”可以组成1~15的数,并且不会出现两种或两种以上的相同情况。
由这个特点,在程序中可以巧妙的使用,目前我发现Android源码中常使用的几个地方:
一、通过Intent Flags对应的值,可以将多种标志通过“或运算”来进行组合, 以下代码是Intent添加标志,使用到“或(|)”运算: 1)
mIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED | Intent.FLAG_ACTIVITY_SINGLE_TOP );
event.mFlags |= FLAG_CANCELED | FLAG_CANCELED_LONG_PRESS;
if ((intent.getFlags()&Intent.FLAG_ACTIVITY_NEW_TASK) == 0){ //条件为真(即等于0),intent.getFlags()不包含NEW_TASK ... }
// 判断该视图是否为disable 状态 这里ENABLED_MASK的值与 DISABLED的值一样 if ((viewFlags & ENABLED_MASK) == DISABLED) { ... }// 返回是否可点击 return (((viewFlags & CLICKABLE) == CLICKABLE || (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE));
mFlags &= ~FLAG_START_TRACKING; // 清除mFlags中的FLAG_START_TRACKING
…… private static final int PRESSED = 0x00004000; int mPrivateFlags ;…… public void setPressed(boolean pressed) { if (pressed) { mPrivateFlags |= PRESSED; // 添加PRESSED状态 } else { mPrivateFlags &= ~PRESSED; // 取消PRESSED状态 } refreshDrawableState(); dispatchSetPressed(pressed); } 附录:位运算主要是直接操控二进制时使用 ,主要目的是节约内存,使你的程序速度更快,还有就是对内存要求苛刻的地方使用,以下是一牛人总结的方法,分享一下:
位运算应用口诀
清零取反要用与,某位置一可用或
若要取反和交换,轻轻松松用异或
包括:
“>> 右移”;“<< 左移”;“>>> 无符号右移”
例子:
-5>>3=-1
1111 1111 1111 1111 1111 1111 1111 1011
1111 1111 1111 1111 1111 1111 1111 1111
其结果与 Math.floor((double)-5/(2*2*2)) 完全相同。
-5<<3=-40
1111 1111 1111 1111 1111 1111 1111 1011
1111 1111 1111 1111 1111 1111 1101 1000
其结果与 -5*2*2*2 完全相同。
5>>3=0
0000 0000 0000 0000 0000 0000 0000 0101
0000 0000 0000 0000 0000 0000 0000 0000
其结果与 5/(2*2*2) 完全相同。
5<<3=40
0000 0000 0000 0000 0000 0000 0000 0101
0000 0000 0000 0000 0000 0000 0010 1000
其结果与 5*2*2*2 完全相同。
-5>>>3=536870911
1111 1111 1111 1111 1111 1111 1111 1011
0001 1111 1111 1111 1111 1111 1111 1111
无论正数、负数,它们的右移、左移、无符号右移 32 位都是其本身,比如 -5<<32=-5、-5>>32=-5、-5>>>32=-5。
一个有趣的现象是,把 1 左移 31 位再右移 31 位,其结果为 -1。
0000 0000 0000 0000 0000 0000 0000 0001
1000 0000 0000 0000 0000 0000 0000 0000
1111 1111 1111 1111 1111 1111 1111 1111
位逻辑运算符
包括:
& 与;| 或;~ 非(也叫做求反);^ 异或
“& 与”、“| 或”、“~ 非”是基本逻辑运算,由此可以演变出“与非”、“或非”、“与或非”复合逻辑运算。“^ 异或”是一种特殊的逻辑运算,对它求反可以得到“同或”,所以“同或”逻辑也叫“异或非”逻辑。
例子:
5&3=1
0000 0000 0000 0000 0000 0000 0000 0101
0000 0000 0000 0000 0000 0000 0000 0011
0000 0000 0000 0000 0000 0000 0000 0001
-5&3=3
1111 1111 1111 1111 1111 1111 1111 1011
0000 0000 0000 0000 0000 0000 0000 0011
0000 0000 0000 0000 0000 0000 0000 0011
5|3=7
0000 0000 0000 0000 0000 0000 0000 0101
0000 0000 0000 0000 0000 0000 0000 0011
0000 0000 0000 0000 0000 0000 0000 0111
-5|3=-5
1111 1111 1111 1111 1111 1111 1111 1011
0000 0000 0000 0000 0000 0000 0000 0011
1111 1111 1111 1111 1111 1111 1111 1011
~5=-6
0000 0000 0000 0000 0000 0000 0000 0101
1111 1111 1111 1111 1111 1111 1111 1010
~-5=4
1111 1111 1111 1111 1111 1111 1111 1011
0000 0000 0000 0000 0000 0000 0000 0100
5^3=6
0000 0000 0000 0000 0000 0000 0000 0101
0000 0000 0000 0000 0000 0000 0000 0011
0000 0000 0000 0000 0000 0000 0000 0110
-5^3=-8
1111 1111 1111 1111 1111 1111 1111 1011
0000 0000 0000 0000 0000 0000 0000 0011
1111 1111 1111 1111 1111 1111 1111 1000