基于BIT数组实现全局功能开关

简介: 某一天巧合打开了sofa-bolt项目,查找部分源码,看到了项目中使用bit数组实现功能开关的特性,感觉这种方式可以借鉴,于是写下这篇文章。

前提



某一天巧合打开了sofa-bolt项目,查找部分源码,看到了项目中使用bit数组实现功能开关的特性,感觉这种方式可以借鉴,于是写下这篇文章。


原理



bit数组的布局如下:

微信截图_20220513210221.png


由于每个bit都可以表示1或者0,刚好对应于开关的ONOFF。只需要定义好每个开关所在的bit数组下标和开关的状态(ON = 1或者OFF = 0),通过判断不同开关下标所在的bit即可判断开关的状态:

微信截图_20220513210227.png


  • 优点:节省空间,理论上只需要占用最多2n位的内存(n为开关的数量,这里考虑扩容,扩容让bit数组长度为原来的2倍)
  • 缺点:暂时没发现


实现



JDK中的bit数组可以直接使用BitSet。首先定义开关定义SwitchDef


public class SwitchConst {
    public static final boolean ON = true;
    public static final boolean OFF = false;
}
@RequiredArgsConstructor
@Getter
public enum SwitchDef {
    /**
     * 启用HTTPS
     */
    ENABLE_HTTPS(0, SwitchConst.ON, "启用HTTPS"),
    /**
     * 启用异步
     */
    ENABLE_ASYNC(1, SwitchConst.OFF, "启用异步"),
    ;
    /**
     * 下标
     */
    private final int index;
    /**
     * 默认状态
     */
    private final boolean defaultStatus;
    /**
     * 描述
     */
    private final String description;
    @Override
    public String toString() {
        return String.format("SwitchDef(name=%s,description=%s)", name(), description);
    }
}
复制代码


接着定义开关接口Switch


public interface Switch {
    /**
     * 启用
     *
     * @param switchDef switchDef
     */
    void turnOn(SwitchDef switchDef);
    /**
     * 关闭
     *
     * @param switchDef switchDef
     */
    void turnOff(SwitchDef switchDef);
    /**
     * 判断状态
     *
     * @param switchDef switchDef
     * @return boolean
     */
    boolean status(SwitchDef switchDef);
}
复制代码


最后编写开关实现BitSetSwitch和客户端代码:


public enum BitSetSwitch implements Switch {
    /**
     * 单例
     */
    X;
    BitSetSwitch() {
        init();
    }
    private final BitSet switches = new BitSet();
    @Override
    public void turnOn(SwitchDef switchDef) {
        switches.set(switchDef.getIndex(), SwitchConst.ON);
    }
    @Override
    public void turnOff(SwitchDef switchDef) {
        switches.clear(switchDef.getIndex());
    }
    @Override
    public boolean status(SwitchDef switchDef) {
        return switches.get(switchDef.getIndex());
    }
    private void init() {
        Stream.of(SwitchDef.values()).forEach(item -> switches.set(item.getIndex(), item.isDefaultStatus()));
    }
    public static void main(String[] args) {
        Switch s = BitSetSwitch.X;
        s.turnOn(SwitchDef.ENABLE_HTTPS);
        s.turnOff(SwitchDef.ENABLE_ASYNC);
        System.out.printf("开关[%s],状态:%s%n", SwitchDef.ENABLE_HTTPS, s.status(SwitchDef.ENABLE_HTTPS));
        System.out.printf("开关[%s],状态:%s%n", SwitchDef.ENABLE_ASYNC, s.status(SwitchDef.ENABLE_ASYNC));
    }
}
复制代码


执行该main方法后控制台输出如下:

开关[SwitchDef(name=ENABLE_HTTPS,description=启用HTTPS)],状态:true
开关[SwitchDef(name=ENABLE_ASYNC,description=启用异步)],状态:false
复制代码


仿真场景(伪代码)如下:


Switch s = BitSetSwitch.X;
String uri = "www.throwx.cn";
String schema = "http";
if (s.turnOn(SwitchDef.ENABLE_HTTPS)){
    schema = "https";
}
HttpClint ch = new DefaultHttpClient();
if (s.turnOn(SwitchDef.ENABLE_ASYNC)){
    ch = new AsyncHttpClient();
}
Result r = ch.executeRequest(schema + uri);
......
复制代码


小结



在阅读一些主流框架源码的时候,可以借鉴一些设计合理的方案应用到自身的日常开发中。


参考资料:


相关文章
|
XML 安全 Java
自定义PMD检测的类型集合(详解)
自定义PMD检测的类型集合(详解)
自定义PMD检测的类型集合(详解)
|
7月前
|
存储
Qt更新组件出现(“要继续此操作,至少需要一个有效且已启用的储存库”)
Qt更新组件出现(“要继续此操作,至少需要一个有效且已启用的储存库”)
389 0
Qt更新组件出现(“要继续此操作,至少需要一个有效且已启用的储存库”)
|
7月前
|
JavaScript 前端开发 Java
若依框架---选中某值 其他值自动回调填充
若依框架---选中某值 其他值自动回调填充
213 0
|
网络架构
ES6中新增的rest剩余参数在函数内部的使用问题
ES6 中引入了 rest 参数(...变量名),用于获取函数内不确定的多余参数,注意只能放在所有参数的最后一个
63 0
|
存储 缓存 安全
[oeasy]python0082_[趣味拓展]控制序列_清屏_控制输出位置_2J
[oeasy]python0082_[趣味拓展]控制序列_清屏_控制输出位置_2J
108 0
|
网络架构 索引
ES6新增扩展:字符串-数值-数组-函数-对象
ES6新增扩展:字符串-数值-数组-函数-对象
91 0
|
JavaScript 索引
数组的扩展和新增方法
数组的扩展和新增方法
99 0
|
开发者
cobrautils 使用反射获取 flag 配置, 支持指针字段
cobrautils 使用反射获取 flag 配置, 支持指针字段
128 0
cobrautils 使用反射获取 flag 配置, 支持指针字段
ES6(函数新增特性)
ES6(函数新增特性) 1.函数参数默认值 没有 y 时,默认就是world 有 y 时,输出值即可 (错误) (C有默认值,正确) 默认值后面不能再有没有默认值的变量  2.作用域 y 取其前面的 x ;而不是 x = test;     x 不在 y 的前面  3.test 参数 将输入参数(离散值)转化成数组  4.扩展运算符  将数组转化成离散值   5.箭头函数 arrow为函数名,第一个 = 后为参数, =>后为返回值,    第一个参数可为空。
934 0