以下红字为小弟自己的一些理解。
Flag理解
在阅读源码的时候经常发现有一些标志属性使用一些位操作来判断是否具有该标志,增加标志或者去除标志。
比如View.java中的
/**
* This view does not want keystrokes. Use with TAKES_FOCUS_MASK when
* calling setFlags.
*/
private static final int NOT_FOCUSABLE = 0x00000000;
/**
* This view wants keystrokes. Use with TAKES_FOCUS_MASK when calling
* setFlags.
*/
private static final int FOCUSABLE = 0x00000001;
为什么要使用16进制呢?为何不使用十进制0,1,2...?原因是因为这样就可以方便的用位运算来表示共同具有某些属性或者直接判断是否具有某个属性。
上面这句话我们稍微来分析一下。拿文件的权限来分析一下,我们知道文件有三个权限:读,写,执行。分别用4 , 2 , 1来表示,这样一来就可以很快速的知道文件所有的权限,比如5表示可读可执行两个权限,7表示可读可写可执行权限。
如果把上面的2进制换成10进制呢?结果就会变得很尴尬。假如上面三个权限分别用1 , 2 , 3表示,3这个就会很尴尬,因为我们不知道这个值表示的是可读可写两个权限还是仅仅一个执行权限。
Android源码中主要针对FLAG的运算有三种
①. 增加属性 "|"
如果需要向flag变量中增加某个FLAG,使用"|"运算符
flag |= XXX_FLAG;
原因: 如果flag变量没有XXX_FLAG,则 | 完后flag对应的位为1,如果有XXX_FLAG,则 | 完后值不会变对应位还是1.
②. 包含属性 "&"
如果需要判断flag变量中是否包含XXX_FLAG,使用"&"运算符
flag & XXX_FLAG != 0 或者 flag & XXX_FLAG = XXX_FLAG
原因: 如果flag变量里包含XXX_FLAG,则&完后flag变量对应的位为1,因为XXX_FLAG的定义保证了只有一位非0,其他位都为0,所以如果是包含的话&运算后值不为0,值为此XXX_FLAG的值,不包含的话值为0.
③.取消属性 "&~"
如果需要取消flag变量的XXX_FLAG, 使用 "&~".
flag &= ~XXX_FLAG;
原因: 先对XXX_FLAG进行取反 则XXX_FLAG原来非0的那一位变为0,则使用&运算符后flag变量非0的那一位变为0,则意味着flag变量不包含XXX_FLAG.
这样做的好处就是可以用一个值表示多种状态。假如这里有四种状态:是否可点击,是否获取焦点。我们只需用flag标记 : 0x1,0x2,0x4,0x8 , 然后保存到一个值里面就行,用以上的运算符就能快速的增加删除以及判断属性了。
使用位移<<操作符
static final int VIEW_STATE_WINDOW_FOCUSED = 1;
static final int VIEW_STATE_SELECTED = 1 << 1;
static final int VIEW_STATE_FOCUSED = 1 << 2;
static final int VIEW_STATE_ENABLED = 1 << 3;
static final int VIEW_STATE_ACTIVATED= 1<< 5;
和
static final int VIEW_STATE_WINDOW_FOCUSED = 0x00000001;
static final int VIEW_STATE_SELECTED = 0x00000002;
static final int VIEW_STATE_FOCUSED = 0x00000004;
static final int VIEW_STATE_ENABLED = 0x00000008;
static final int VIEW_STATE_ACTIVATED= 0x00000010;
两种写法是等价的。