Java使用位域进行多标记(状态)管理

Android中位域的应用

在Android中,我们会经常用到或者看到以下这样的代码 :

public class ExampleUnitTest {
    @Test
    public void gravityTest(LayoutParams params) {
        // 视图在layout中右下角显示
        params.gravity = Gravity.RIGHT | Gravity.BOTTOM;
    }
    @Test
    public void intentFlagTest(Intent intent) {
        // 清空任务栈中所有旧的activity
        intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK
                | Intent.FLAG_ACTIVITY_CLEAR_TASK);

        // 如果activity已存在于栈中,清空该activity之上的所有任务
        intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP
                | Intent.FLAG_ACTIVITY_CLEAR_TOP);
    }
    @Test
    public void windowMangerFlags(WindowManager.LayoutParams params) {
        // 不拦截视图以外的事件,在锁屏中显示
        params.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
                | WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED;
    }
}

通过一个 int字段,来添加多个 标志或者状态. 一个int字段,能够管理多个标记(状态)值.
如此神奇的操作怎样实现的呢? 答案就是通过位运算来实现.

位操作基础

java中提供的基础位运算符有 与(&),或(|),非(~),异或(^),左移<<,
右移(>>)无符号右移(>>>).

除了位非(~)是一元操作符外,其它的都是二元操作符。

下面只介绍本文中,使用到的位操作

  • 按位与

A & B : A和B对应的二进制数位都为1时,结果才为1,其他情况为0.

A     =  001101 // 13
B     =  100101 // 37
A & B =  000101 // 5
  • 按位或

A | B : A和B对应的二进制数位都为0时,结果才为0,其他情况为1.

A     =  001101 // 13
B     =  100101 // 37
A | B =  101101 // 45
  • 按位非

~A : 将a的二进制表示每一位进行取反操作,0变1,1变0.
相当于相反数 - 1

A     =  001101 // 13
~A    =  11111111111111111111111111110010 // int32位,补码表示,第一位为符号位
// 根据上诉补码转原码为
//       10000000000000000000000000001110 // -14
  • 左移操作

A << B:将A的二进制表示的每一位向左移B位,左边超出的位截掉,右边不足的位补0。
在取值范围内,移动一位相当于乘2.

A     =  001101 // 13
A << 1 = 011010 // 26

原理与实践

通常情况下,如果多个状态或者标记相互之间有关联, 如布局方向,上下左右,左上,居中 ... 等.我们可能会为每一个标记设置一个变量.

boolean left = false;
boolean right = false;
...
void setLeft(boolean b);
void setRight(boolean b);
...

这种情况下, 有4个标记相互关联,并且能产生新的标记,那么我们就需要设置4个标记变量,然后只能通过一系列的set方法来转换状态.如

v.setLeft(true);
v.setRight(true);

这样就会使得,各个状态不易维护和判断,状态越多,情况越复杂,代码会显得冗长难以维护.

像这种,独立状态(标记)之间相互组合可以产生新的状态(标记),且每个独立状态(标记)只有true或者false值的,我们可以使用位域的概念来管理这些状态.

它的核心思想就是将, int 数值看做是 二进制数位表示.如果有四个状态就可以像这样 0000,用四位二进制表示,每一个二进制位都可以表示一种状态. 然后通过 位运算,来提取或添加标记位.四位对应的组合状态有16个. 而我们,只需要通过一个int变量就能够管理这些状态.

当参与的状态(标记)越多时,如果使用单独的标记变量,就需要生成越多的变量,而用位域,这种独立状态为不管有多少个,都可以用一个变量表示.int类型最多存放32个独立状态

下面我们来看具体实现.(简单的模仿Gravity类的一部分功能)

public class Gravity {
    // 二进制表示 0001
    public static final int LEFT              = 1;
    // 二进制表示 0010
    public static final int RIGHT             = LEFT << 1;
    // 二进制表示 0100
    public static final int TOP               = LEFT << 2;
    // 二进制表示 1000
    public static final int BOTTOM            = LEFT << 3;
    // 水平居中, 二进制表示 0011
    public static final int HORIZONTAL_CENTER = LEFT | RIGHT;
    // 垂直居中, 二进制表示 1100
    public static final int VERTICAL_CENTER   = TOP | BOTTOM;
    // 居中, 二进制表示 1111
    public static final int CENTER            = HORIZONTAL_CENTER | VERTICAL_CENTER;
    // 默认左上角, 二进制表示 0101
    public static final int DEFAULT           = LEFT | TOP;

    // 存放标志位
    private int mFlags = DEFAULT;

    // 设置标记位,会清除原来的标记
    public void setFlags(int flags) {
        mFlags = flags;
    }

    // 添加标记位,在原来的基础上添加
    public void addFlags(int flags) {
        mFlags |= flags;
    }

    // 清除指定的标记
    public void clearFlags(int flags) {
        mFlags &= ~flags;
    }

    // 清除所有标记,设为默认
    public void clears() {
        mFlags = DEFAULT;
    }

    // 判断是否存在指定的标记
    public boolean hasFlags(int flags) {
        return (mFlags & flags) == flags;
    }

    // 判断是否 只有指定的标记
    public boolean onlyFlags(int flags) {
        return mFlags == flags;
    }

    public void apply() {
        String des = "左上角";
        if (hasFlags(CENTER)) {
            des = "整体居中";
        } else if (hasFlags(HORIZONTAL_CENTER)) {
            if (hasFlags(BOTTOM)) des = "水平居中,竖直向下";
            else des = "水平居中,竖直向下";
        } else if (hasFlags(VERTICAL_CENTER)) {
            if (hasFlags(RIGHT)) des = "竖直居中,水平向右";
            else des = "竖直居中,水平向左";
        } else if (hasFlags(LEFT | BOTTOM)) {
            des = "左下角";
        } else if (hasFlags(RIGHT | TOP)) {
            des = "右上角";
        } else if (hasFlags(RIGHT | BOTTOM)) {
            des = "右下角";
        }
        System.out.println("你选择的布局是 : " + des);
    }
}

具体的调用实现 :


public class Main {
    public static void main(String[] args) {
        Gravity gravity = new Gravity();

        // 设置为 右下角
        gravity.setFlags(Gravity.BOTTOM | Gravity.RIGHT);
        gravity.apply();

        // 添加 left后,变为 水平居中,竖直向下
        gravity.addFlags(Gravity.LEFT);
        gravity.apply();

        // 判断是否 水平居中, 返回为 true
        gravity.hasFlags(Gravity.HORIZONTAL_CENTER);

        // 添加top后,变为 整体居中了
        gravity.addFlags(Gravity.TOP);
        gravity.apply();

        // 删掉 bottom和left 之后,变为右上角了
        gravity.clearFlags(Gravity.BOTTOM | Gravity.LEFT);
        gravity.apply();
    }
}
// 结果
你选择的布局是 : 右下角
你选择的布局是 : 水平居中,竖直向下
你选择的布局是 : 整体居中
你选择的布局是 : 右上角
上一篇:看不见≠没干 浅述你所不了解的英特尔大数据


下一篇:详解开源大数据引擎Greenplum的架构和技术特点