对于 Java 类应用,内存方面需要注意:

  1. 不要占用大量内存,否则可用内存少;触发 GC 或 OutOfMemoryError

  2. 不要频繁创建对象,频繁内存分配,触发 GC。

对于枚举和常量:

  1. 使用枚举,并不会使得对象的创建更加频繁。

  2. 枚举类会比常量占用更多的内存,在程序运行期间,如果不卸载枚举类,内存就一直占用着。

    相对于常量,枚举占用的内存是较为可观的。

使用常量,可以大量节省内存,在 C 之类的语言中,大量使用 BitMask 来进行状态表示。

在 Android 中,也大量地使用了 BitMask,比如 android.view.View 这个类。

位操作

在使用 BitMask 前,我们先复习一下基本的位操作。

  1. NOT

    NOT 0000 0001
    = 1111 1110

    比如:

    int a = 1;
    int b = ~a;
  2. OR

    OR  0000 0001
    0000 0010
    = 0000 0011

    比如:

    int a = 1;
    int b = 2;
    int c = a | b;
  3. AND

    AND 0000 0101
    0000 0110
    = 0000 0100

    比如:

    int a = 5;
    int b = 6;
    int c = a & b;

BitMask

我们知道,每一个 bit 可以有两种取值:0 或 1。

BitMask 采用一个数值来记录状态,使用这个数值的每一位来表达一个状态。

使用 BitMask 可用非常少的资源表达非常丰富的状态。

在 Java 中,一个 byte 类型,有 8 位(bit),可以表达 8 个不同的状态,并且这些状态是互不影响的。而 int 类型,则有 32 位,可以表达 32 种状态。

更为重要的是,基于 BitMask 可 非常简单地 进行组合状态查询。

BitMask 基本操作

假设我们用一个表示状态的数值: status,初始值为 0。

byte status = 0;

我们定义一个 mask 数值,该数第二位为 1:0000 0010

我们把 1 往左移动 1 位来得到这个数:

byte mask = 0x01 << 1;
  1. 设置状态

    其他位不管,把第 2 位变为 1 即可。

        xxxx xxxx
    OR 0000 0010
    = xxxx xx1x

    代码

    status |= mask;
  2. 清除状态

    其他位不管,把第 2 位置为 0。

        xxxx xxxx
    AND 1111 1101
    = xxxx xx0x

    这实际是对 status 和 mask 的反码进行逻辑『与』运算:

    status &= ~mask;
  3. 查询状态

    确定第 2 位是 0 还是 1,和 mask 进行逻辑『与』运算:

        xxxx xxxx
    AND 0000 0010
    = 0000 00x0

    如果为 1,返回一个大于 0 的值,否则返回 0。

    boolean isOn = (status & mask) > 0;

例子

下面结合一个例子来做说明。

相关代码在这里: https://github.com/liaohuqiu/android-BitMaskSample

李白 是个诗人,生活简单『朴素』:有时候写诗;有时候喝酒;有时候边写诗,边喝酒。

不管是忙于写诗还是忙于喝酒,李白都是在忙碌状态中。

我们用一个字节来表示他的状态,一个字节有 8 位,我们从低位起开始取两位分别代表写诗和喝酒。

   writing  ------+
|
v
-------+---+---+---+
x x | | | x |
-------+---+---+---+
^
|
drinking --+

两个 mask 为:

// 0000 0010
private static final byte STATE_BUSY_IN_WRITING = 0x01 << 1; // 0000 0100
private static final byte STATE_BUSY_IN_DRINKING = 0x01 << 2;
  1. 状态设置与清除

    以设置 drinking 状态为例子,设置状态即和 mask 值进行逻辑『或』,清除状态与 mask 的反码进行逻辑『与』运算。

    public void setBusyInDrinking(boolean busy) {
    if (busy) {
    mState |= STATE_BUSY_IN_DRINKING;
    } else {
    mState &= ~STATE_BUSY_IN_DRINKING;
    }
    }
  2. 状态查询

    与 mask 进行逻辑『与』运行,判断是否为零即可:

    public boolean isBusyInDrinking() {
    return (mState & STATE_BUSY_IN_DRINKING) != 0;
    }
  3. 组合状态查询

    不管是忙于写诗还是忙于饮酒,都称为『李白很忙』,这是一种组合状态。只要处于这两种状态中的一种,即处于组合状态中。

    要进行状态组合,用逻辑『或』运算即可,当进行多个状态组合时,特别方便:

    STATE_BUSY_MASK = STATE_BUSY_IN_WRITING | STATE_BUSY_IN_DRINKING

    判断是否处于组合状态中:

    public boolean isBusy() {
    return (mState & STATE_BUSY_MASK) != 0;
    }

Android 中的 IntDef

使用 IntDef 注解来声明常量值,定义变量时,加上 IntDef 所定义的声明,编译器会检查赋值是否合法。

声明:

// 最后 8 位 0000 1100
static final int VISIBILITY_MASK = 0x0000000C; public static final int VISIBLE = 0x00000000; // 最后 8 位 0000 0100
public static final int INVISIBLE = 0x00000004; // 最后 8 位 0000 1000
public static final int GONE = 0x00000008; @IntDef({VISIBLE, INVISIBLE, GONE})
@Retention(RetentionPolicy.SOURCE)
public @interface Visibility {}

使用:

public void setVisibility(@Visibility int visibility) {
setFlags(visibility, VISIBILITY_MASK);
}

上面我们看到,代码中采用了最左的 3,4 位来表达 View 的可见性。

结论

除了 IntDef,还有 StringDef,有兴趣的同学可以看源码。

在 Android 的代码中有大量的 BitMask 的运用,像 ViewMotionEvent 这样的核心基础类中,需要认真考虑内存的使用,能省则省。

如果你真想完全地掌控内存的使用,追求卓越的品质,想最大限度节省内存,BitMask 是你不错的选择。

同时,我们也应该清楚枚举也不是不能用。

我听到过很多论调,说用『枚举不好,官方也建议别用,因为占用很多内存,效率不高』,这些也都是人云亦云的典型。

实际上,除非你写的是类似 View 这样的核心基础类或者超大型应用,否则,如果连枚举这样内存开销都有问题的话,这个项目的问题就真的大了。

BitMask 使用参考的更多相关文章

  1. Hyper-v 安装CentOS 7 (其他虚拟机一样参考)

    平台之大势何人能挡? 带着你的Net飞奔吧!http://www.cnblogs.com/dunitian/p/4822808.html hyper-v安装很多人没弄过,我这里介绍一下.(其他虚拟机参 ...

  2. ifconfig: command not found(CentOS专版,其他的可以参考)

    ifconfig: command not found 查看path配置(echo相当于c中的printf,C#中的Console.WriteLine) echo $PATH 解决方案1:先看看是不是 ...

  3. 参考bootstrap中的popover.js的css画消息弹框

    前段时间小颖的大学同学给小颖发了一张截图,图片类似下面这张图: 小颖当时大概的给她说了下,其实小颖也不知道上面那个三角形怎么画嘻嘻,给她说了DOM结构,具体的css让她自己百度,今天小颖自己参考boo ...

  4. Oracle安装部署,版本升级,应用补丁快速参考

    一.Oracle安装部署 1.1 单机环境 1.2 Oracle RAC环境 1.3 Oracle DataGuard环境 1.4 主机双机 1.5 客户端部署 二.Oracle版本升级 2.1 单机 ...

  5. Angularjs参考框架地址

    1.Table(Grid)参考地址 https://github.com/samu/angular-table https://github.com/daniel-nagy/md-data-table ...

  6. iOS开发之三个Button实现图片无限轮播(参考手机淘宝,Swift版)

    这两天使用Reveal工具查看"手机淘宝"App的UI层次时,发现其图片轮播使用了三个UIButton的复用来实现的图片循环无缝滚动.于是乎就有了今天这篇博客,看到“手机淘宝”这个 ...

  7. SQL 性能调优中可参考的几类Lock Wait

    在我们的系统出现性能问题时,往往避不开调查各种类型 Lock Wait,如Row Lock Wait.Page Lock Wait.Page IO Latch Wait等.从中找出可能的异常等待,为性 ...

  8. RMAN异机恢复快速参考

    应用场景:服务器A为正常运行的生产环境,需要在服务器B上部署一套相同环境做测试. 数据库环境:RHEL6.4 + Oracle 11.2.0.4.7 一. 服务器A备份数据库 1.1 在线备份(数据库 ...

  9. Linux平台oracle 11g单实例 安装部署配置 快速参考

    1.重建主机的Oracle用户 组 统一规范 uid gid 以保证共享存储挂接或其他需求的权限规范 userdel -r oracle groupadd -g 7 oinstall groupadd ...

随机推荐

  1. SQLserver利用系统时间生成“2015-11-30 00:00:00.000”类型的时间

    select getdate() ---当前时间:2015-12-18 10:20:24.097 -------------------建立测试表 Create Table #Test ( ID IN ...

  2. using System.Threading.Tasks;

    using System.Threading.Tasks; .Net并行库介绍——Task1

  3. 使用parseJSON代替eval

    有些程序员如果没有很好的在javascript中解析json数据,往往会直接eval把json转成js对象,这时候如果json的数据中包含了被注入的恶意数据,则可能导致代码注入的问题. 正确的做法是分 ...

  4. bnuoj 4208 Bubble sort

    http://www.bnuoj.com/bnuoj/problem_show.php?pid=4208 [题意]:如题,求冒泡排序遍历趟数 [题解]:这题开始2B了,先模拟TLE,然后想了一下,能不 ...

  5. 3144:[HNOI2013]切糕 - BZOJ

    题目描述 Description 经过千辛万苦小 A 得到了一块切糕,切糕的形状是长方体,小 A 打算拦腰将切糕切成两半分给小 B.出于美观考虑,小 A 希望切面能尽量光滑且和谐.于是她找到你,希望你 ...

  6. FACL的使用

    ACL的使用 ACL即Access Control List 主要的目的是提供传统的owner,group,others的read,write,execute权限之外的具体权限设置,ACL可以针对单一 ...

  7. ios 多任务学习笔记

    一.检测多任务是否支持: - (BOOL) isMultitaskingSupported{ BOOL result = NO; if ([[UIDevice currentDevice] respo ...

  8. 把只包含因子2、3和5的数称作丑数(Ugly Number)。例如6、8都是丑数,但14不是,因为它包含因子7。 习惯上我们把1当做是第一个丑数。求按从小到大的顺序的第N个丑数。

    // ConsoleApplication1.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> ...

  9. 【BZOJ】【1927】【SDOI2010】星际竞速

    网络流/费用流 比较简单的一题,对于每个星球,将它拆成两个点,然后二分图建模:左部结点与S相连,流量为1费用为0:右部结点与T相连,流量为1费用为0:对于每条航道x->y,连边x->y+n ...

  10. SqlBulkCopy 简单运用

    using(SqlConnection conn = new SqlConnection(str)) { conn.Open(); using (System.Data.SqlClient.SqlBu ...