你是否留意过“位运算”

长时间阅读Android SDK源码,会发现Google喜欢用位运算,伴随的是代码中会定义一堆int类型的常量,乍一看很懵逼,特别是所在View相关的类里边,比如这些常量你可熟悉:

    static final int FLAG_CLIP_CHILDREN = 0x1;

    private static final int FLAG_CLIP_TO_PADDING = 0x2;

    static final int FLAG_INVALIDATE_REQUIRED  = 0x4;

    private static final int FLAG_RUN_ANIMATION = 0x8;

    static final int FLAG_ANIMATION_DONE = 0x10;

    private static final int FLAG_PADDING_NOT_NULL = 0x20;

    private static final int FLAG_ANIMATION_CACHE = 0x40;

    static final int FLAG_OPTIMIZE_INVALIDATE = 0x80;

    static final int FLAG_CLEAR_TRANSFORMATION = 0x100;

又或者这种运算算,乍一看也不知道标识啥意思:

   // This is the original call.
            try {
                mGroupFlags |= FLAG_START_ACTION_MODE_FOR_CHILD_IS_NOT_TYPED;
                return startActionModeForChild(originalView, callback, ActionMode.TYPE_PRIMARY);
            } finally {
                mGroupFlags &= ~FLAG_START_ACTION_MODE_FOR_CHILD_IS_NOT_TYPED;
            }

Google常用套路:

1.int类型通过位运算存储boolean

下面我写个java文件,你可能一看就明白

public class Youyisi {

    private static final int BOOL01 = 1;
    private static final int BOOL02 = 1<<1;
    private static final int BOOL03 = 1<<3;
    private static final int BOOL04 = 1<<4;
    private static final int BOOL05 = 1<<5;
    private static final int BOOL06 = 1<<6;
    private static final int BOOL07 = 1<<7;
    private static final int BOOL08 = 1<<8;
    //most << 31
    

    private int flag;

    public void setBool01(boolean b){

        setBOOL(b, BOOL01);

    }
    public void setBool02(boolean b){

        setBOOL(b, BOOL02);

    }
    public void setBool03(boolean b){
        setBOOL(b, BOOL03);

    }
    public void setBool04(boolean b){
        setBOOL(b, BOOL04);

    }


    public boolean getBool1(){
        return getBOOL(BOOL01);
    }

    public boolean getBool2(){
        return getBOOL(BOOL02);
    }
    public boolean getBool3(){
        return getBOOL(BOOL03);
    }
    public boolean getBool4(){
        return getBOOL(BOOL04);
    }


    private boolean getBOOL(int FLAG){
        return (flag & FLAG) != 0;
    }


    private void setBOOL(boolean b,int FLAG){
        if (b == ((flag & FLAG) != 0)) {

            return;
        }
        if (b) {
            flag |= FLAG;
        } else {
            flag &= ~FLAG;
        }
    }

    public static void main(String[] args){
        Youyisi y = new Youyisi();
        y.setBool01(true);
        y.setBool02(true);
        y.setBool03(true);
        y.setBool04(true);
        boolean b  = y.getBool4();
        System.out.println(b);
    }
    
}

java中,int类型长度是32位,化成二进制,每一位上是0、1与否,都能代表一个boolean,所以一个int类型可以存储32个boolean值;
setBool过程分析:如果要给FLAG名下设置true,要做的事就是把flag对的FLAG有效位置1,所以就进行 flag |= FLAG运算;相反设置0,就进行 flag &= ~FLAG运算;
getBool过程分析:如果flag和FLAG按位想与结果为1,则为true,否则为false,所以采取(flag & FLAG) != 0;

2.多个相关联的int/bool/enum,通常按位存储在一个int类型中

什么意思呢,比如每一个都有性别和年龄,假设性别有男、女、未知三种,年龄是int类型的数值,最大值是300,那么就可以把性别和年龄存储在一个int类型中,这只是我简单的举例,看看Android自定义控件中有个MeasureSpec类,是如何将测量模式和size用同一个int表示;

常量的定义

        private static final int MODE_SHIFT = 30;
        private static final int MODE_MASK  = 0x3 << MODE_SHIFT;

        /**
         * Measure specification mode: The parent has not imposed any constraint
         * on the child. It can be whatever size it wants.
         */
        public static final int UNSPECIFIED = 0 << MODE_SHIFT;

        /**
         * Measure specification mode: The parent has determined an exact size
         * for the child. The child is going to be given those bounds regardless
         * of how big it wants to be.
         */
        public static final int EXACTLY     = 1 << MODE_SHIFT;

        /**
         * Measure specification mode: The child can be as large as it wants up
         * to the specified size.
         */
        public static final int AT_MOST     = 2 << MODE_SHIFT;

构造方法就是合并的操作

 public static int makeMeasureSpec(@IntRange(from = 0, to = (1 << MeasureSpec.MODE_SHIFT) - 1) int size,
                                          @MeasureSpecMode int mode) {
            if (sUseBrokenMakeMeasureSpec) {
                return size + mode;
            } else {
                return (size & ~MODE_MASK) | (mode & MODE_MASK);
            }
        }

分别获取模式和尺寸

//获取模式
 public static int getMode(int measureSpec) {
            //noinspection ResourceType
            return (measureSpec & MODE_MASK);
        }

//获取尺寸
 public static int getSize(int measureSpec) {
            return (measureSpec & ~MODE_MASK);
        }

分析:MODE_MASK=0x3 << MODE_SHIFT即把0x3向左移动30位,得到结果就是0x11000....000,一个30个0;
合并过程分析:如果sUseBrokenMakeMeasureSpec,sUseBrokenMakeMeasureSpec就是size和mode是做过运算的,得到的值是
直接相加size + mode,否则将size和mode坐位运算,得到的值高2位代码mode,低30位代表size;
获取的分析:获取分析比较简单,就直接分别获取高2位和低30位;


上面两个例子说明按位运算的强大之处,但是Java程序员似乎不习惯使用,因为习惯了面向对象,可能会在类中多定义几个属性,那样看起来还简单明了,何必绕来绕去的;但是位操作是不是略显逼格,就是cool,虽然不确定这样能优化多少,但google既然这样做,我们可以尝试追随;

总结:

其实这些都是计算机基础知识,只是上学时觉得没地方用,工作后也没有留意,至于要不要这样用,又是见仁见智的时候了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容

  • 一、基本数据类型 注释 单行注释:// 区域注释:/* */ 文档注释:/** */ 数值 对于byte类型而言...
    龙猫小爷阅读 4,257评论 0 16
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,598评论 18 399
  • 2.17.6.2 星期五 晴 今天接孩子,老师反映,孩子最近态度不怎么好,有点溺爱了。我一听就知道...
    张萌张迪妈妈阅读 165评论 0 0
  • 爱情是什么,谁也说不好。就像前段时间的电影《嫌疑人X的献身》,解开这题X的就是爱情。对于爱情的理解,中国人喜欢写聊...
    小奶茶2333阅读 501评论 2 4
  • 1.速度 = 距离/ 时间.距离至少包含两点:有起点和终点.如果没有两点,就没有距离,也就没有速度,没有变化.时间...
    95d48fb8d89d阅读 231评论 0 0