我们在看安卓源码的时候经常看到类似

flags |= PackageManager.MATCH_DIRECT_BOOT_AWARE | PackageManager.MATCH_DIRECT_BOOT_UNAWARE

这样的代码,让人一头雾水。

那么这个代码到底是什么意思呢?让我们来一步一步解读

比如我们拿Intent源码来说

我们按数值大小给他排个序如下:public static final int FLAG_ACTIVITY_NEW_TASK                        = 0x10000000;
public static final int FLAG_ACTIVITY_MULTIPLE_TASK                   = 0x08000000;
public static final int FLAG_ACTIVITY_CLEAR_TOP                       = 0x04000000;
public static final int FLAG_ACTIVITY_FORWARD_RESULT                  = 0x02000000;
public static final int FLAG_ACTIVITY_PREVIOUS_IS_TOP                 = 0x01000000;
public static final int FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS            = 0x00800000;
public static final int FLAG_ACTIVITY_BROUGHT_TO_FRONT                = 0x00400000;
public static final int FLAG_ACTIVITY_RESET_TASK_IF_NEEDED            = 0x00200000;
public static final int FLAG_ACTIVITY_LAUNCHED_FROM_HISTORY           = 0x00100000;
public static final int FLAG_ACTIVITY_CLEAR_WHEN_TASK_RESET           = 0x00080000;
public static final int FLAG_ACTIVITY_NO_USER_ACTION                  = 0x00040000;
public static final int FLAG_ACTIVITY_REORDER_TO_FRONT                = 0X00020000;
public static final int FLAG_ACTIVITY_NO_ANIMATION                    = 0X00010000;
public static final int FLAG_ACTIVITY_CLEAR_TASK                      = 0X00008000;
public static final int FLAG_ACTIVITY_TASK_ON_HOME                    = 0X00004000;
public static final int FLAG_ACTIVITY_RETAIN_IN_RECENTS               = 0x00002000;
public static final int FLAG_ACTIVITY_LAUNCH_ADJACENT                 = 0x00001000;
public static final int FLAG_ACTIVITY_MATCH_EXTERNAL                  = 0x00000800;
public static final int FLAG_ACTIVITY_REQUIRE_NON_BROWSER             = 0x00000400;
public static final int FLAG_ACTIVITY_REQUIRE_DEFAULT                 = 0x00000200;

我们知道,32位系统里面最大的整型是2^32  ,转换成十六进制数就是0x100000000

也就是一个9位的十六进制数,但是这个9位的十六进制数除了最高位是1外其他的位只能是0,否则就溢出。所以实际上能表达的只有8位十六进制数。所以我们看到Intent里面关于Activity的FLAG都是8位十六进制数。

而且通过排列之后,我们发现每一个FLAG只在8位中其中一位有值,其他7位全是0。而且每一个有值的位上全是1、2、4、8。我们可以看出这个是2的升序次幂。

我们把上面的数值表换算成二进制并排序看一下

0x10000000  = 10000000000000000000000000000
0x08000000  = 01000000000000000000000000000
0x04000000  = 00100000000000000000000000000
0x02000000  = 00010000000000000000000000000
0x01000000  = 00001000000000000000000000000
0x00800000  = 00000100000000000000000000000
0x00400000  = 00000010000000000000000000000
0x00200000  = 00000001000000000000000000000
0x00100000  = 00000000100000000000000000000
0x00080000  = 00000000010000000000000000000
0x00040000  = 00000000001000000000000000000
0X00020000  = 00000000000100000000000000000
0X00010000  = 00000000000010000000000000000
0X00008000  = 00000000000001000000000000000
0X00004000  = 00000000000000100000000000000
0x00002000  = 00000000000000010000000000000
0x00001000  = 00000000000000001000000000000
0x00000800  = 00000000000000000100000000000
0x00000400  = 00000000000000000010000000000
0x00000200  = 00000000000000000001000000000

可以看出每一个Flag值都在一个二进制数指定的位上,这个位是1则表示有这个Flag,这个位上为0则表示没有这个Flag。

为什么要这样做呢?

1、首先,这样做可以用有限的标志位表达非常多样的状态组合,这在我们记录一件比较事物复杂多样的状态的时候,可以大量减少数据量。

比如我们想要表达某个状态同时拥有N个标志,那么我们只要把这N个标志按位或,就可以得到一个唯一值。Flag是29位二进制数,可以有29种标识。那么理论上可以表达这29种标志组合后的 2^29 = 536870912种状态。

2、其次,这样非常有利于数据存储和传输

我们在传输某件事物的这29种Flag各自状态的时候,只需要传输这29种Flag的按位或值。接收方也只存储这一个值,在解读的时候如果要判断是否拥有这个Flag,只要将这个值和Flag按位与不等于0,就说明拥有这个Flag。

总结:不得不感叹这样设计的精妙,29种标志的各自状态只需要1个值就可以表达,可以表达的值范围达到[0,536870912]。而且存储和解读都非常简单高效,有点类似于数据加密压缩了,效率反而提升了。

彻底理解android的各种Flag以及|运算相关推荐

  1. 深入理解Android中View

    文章目录 [隐藏] 一.View是什么? 二.View创建的一个概述: 三.View的标志(Flag)系统 四.MeasureSpec 五.几个重要方法简介 5.1 onFinishInflate() ...

  2. java中view是什么_深入理解Android中View

    文章目录 [隐藏] 这回我们是深入到View内部,去研究View,去了解View的工作,抛弃其他因素,以便为以后能灵活的使用自定义空间打下一定的基础.希望有志同道合的朋友一起来探讨,深入Android ...

  3. 理解Android Binder机制(1/3):驱动篇

    Binder的实现是比较复杂的,想要完全弄明白是怎么一回事,并不是一件容易的事情. 这里面牵涉到好几个层次,每一层都有一些模块和机制需要理解.这部分内容预计会分为三篇文章来讲解.本文是第一篇,首先会对 ...

  4. 《深入理解Android 卷III》第四章 深入理解WindowManagerService

    <深入理解Android 卷III>即将公布,作者是张大伟.此书填补了深入理解Android Framework卷中的一个主要空白.即Android Framework中和UI相关的部分. ...

  5. 理解 Android 消息机制

    本人只是Android小菜一个,写技术文章只是为了总结自己最近学习到的知识,从来不敢为人师,如果里面有不正确的地方请大家尽情指出,谢谢! 本文基于原生 Android 9.0 源码来解析 Androi ...

  6. 细读《深入理解 Android 内核设计思想》(四)Binder 机制 [中]

    对冗余挑拣重点,对重点深入补充,输出结构清晰的精简版 智能指针 binder 驱动中的结构体 ProcessState IPCThreadState BpBinder BinderProxy Serv ...

  7. 《深入理解Android 卷III》第八章深入理解Android壁纸(完整版)

    第8章 深入理解Android壁纸 本章主要内容: ·  讨论动态壁纸的实现. ·  在动态壁纸的基础上讨论静态壁纸的实现. ·  讨论WMS对壁纸窗口所做的特殊处理. 本章涉及的源代码文件名及位置: ...

  8. [深入理解Android卷一全文-第三章]深入理解init

    由于<深入理解Android 卷一>和<深入理解Android卷二>不再出版,而知识的传播不应该因为纸质媒介的问题而中断,所以我将在CSDN博客中全文转发这两本书的全部内容. ...

  9. [深入理解Android卷一全文-第五章]深入理解常见类

    由于<深入理解Android 卷一>和<深入理解Android卷二>不再出版,而知识的传播不应该因为纸质媒介的问题而中断,所以我将在CSDN博客中全文转发这两本书的全部内容. ...

  10. 深入理解Android相机体系结构之六

    该系列文章总目录链接: https://blog.csdn.net/u012596975/article/details/107135938 本篇是<深入理解Android相机体系结构>连 ...

最新文章

  1. Maven使用笔记(四)pom.xml配置详解
  2. [译]深度神经网络的多任务学习概览(An Overview of Multi-task Learning in Deep Neural Networks)...
  3. AutoCAD如何输入文字
  4. 小明分享|ESP8266设置STA模式ping包测试
  5. 与其感慨路难行,不如马上出发
  6. 编译32位_玩转Android10源码开发定制(11)内核篇之安卓内核模块开发编译
  7. 活动:北京Xamarin分享会第8期(2017年11月11日)
  8. 为什么用scrum_为什么Scrum糟糕于数据科学
  9. 计算机文档xsl,XSL-FO 文档
  10. linux怎么安装Nginx
  11. python中abs函数是什么意思_实例讲解Python3中abs()函数
  12. ICPC 昆明 单挑打铁记
  13. VMware虚拟机Ubuntu自适应屏幕大小
  14. 怎样搭建企业内部wiki
  15. java替换一个反斜杠_java反斜杠替换
  16. HTML Parser Jsoup - 网页抓取百度百科信息的例子
  17. IDEA中同窗口导入新的maven项目
  18. 软文怎么做好内容营销
  19. 【转自知乎】5G关键技术之波束成形
  20. JS正则表达式写法:手机号、邮箱、用户只能由英文字母或数字组成,长度为4-16个字符,并且以英文字母开头

热门文章

  1. 传统架构 vs 云原生架构,谈谈为什么我们需要云原生架构?
  2. 字符串与整型的相互转换
  3. java html截图_Java实现网页截图/登录截图
  4. 洛克人java下载_洛克人5-布鲁斯的阴谋
  5. 51单片机 1-LED灯流水灯 练习3-流水灯(_crol_位移)
  6. IE无法查看源文件原因及应用技巧
  7. Google - 搜索图片快速设置
  8. 元宇宙大杀器来了!小扎祭出4款VR头显,挑战视觉图灵测试
  9. 关于el-dialog取消事件
  10. php换行快捷输入,excel中换行的快捷键是什么