1.BitMap介绍
Redis 的位图(bitmap)是由多个二进制位组成的数组,只有两种状态,0和1, 数组中的每个二进制位都有与之对应的偏移量(从 0 开始),通过这些偏移量可以对位图中指定的一个或多个二进制位进行操作。
2.Redisson使用
2.1 bitmap的获取和设置
2.1.1 基础方法
getBitSet方法,获取对应键的bitMap缓存。set(offset)方法,设置对应offset位置值为1,bitMap中用true表示1,false表示0。
// RBitSet getBitSet(String name);
RBitSet bitTest = redissonClient.getBitSet("bitTest");
// 如果bitTest键不存在,直接调用length方法会报错
// System.out.println(bitTest.length());
System.out.println(bitTest.size());
bitTest.set(3);
System.out.println(bitTest.length());
System.out.println(bitTest.size());
打印结果:
0
4
8
redis 缓存value:
00010000
通过结果,可以发现:bitmap的size是byte的整数倍,我们只需要设置为1的bit位,其他位置会自动补0;length方法返回的是最后一个bit为1的长度,size返回的是整个bitmap的长度。我们统计数据时,如果需要标记上次统计到哪一位,我们可以在每次统计后,都将下一bit置为1,然后通过length获取长度,就可以知道上次统计到哪一位了。
对于,key是否存在,可以通过isExists方法判断。在设置过期时间时,要注意,如果key不存在的话,要先set值,然后再设置过期时间,不然不生效。
if (!bitTest.isExists()){
bitTest.set(3);
bitTest.expire(30, TimeUnit.SECONDS);
}
我们还可以通过cardinality方法,获取bit位为1的数量。
long cardinality = bitTest.cardinality();
2.1.2 操作byte\short\int\long
对于byte\short\int\long提供了对应的set和get方法,区别就是对应的字节长度不一样,用bit位表示即8bit\16bit\32bit\64bit。
以int类型的api为例:
int getInteger(long offset);
int setInteger(long offset, int value);
offset为3,即起始位index为3,写一个2进制的int数字10,即32位,由于bitMap的size是byte的整数倍,需要补全为5byte,长度应该是:3bit + 32bit + 5bit。所以,我们在getInteger的时候,如果offset为3,则获取的结果是10,如果<3,则相当于左移,值缩小2的整数倍,>3,则相当于右移,值扩大2的整数倍。不过注意的是,后面如果再次set的时候会重置重复的bit位。比如,我在offset为3的位置set一个long值,即64位,这64位就会被重置。
@Test
public void testRedisson(){
RBitSet bitTest = redissonClient.getBitSet("bitTest");
bitTest.setInteger(3,10);
// 0000000000000000000000000000000101000000
System.out.println(bitTest.getInteger(3));
System.out.println(bitTest.getInteger(2));
bitTest.setLong(3,12);
// 000000000000000000000000000000000000000000000000000000000000000110000000
System.out.println(bitTest.getLong(3));
System.out.println(bitTest.getLong(4));
System.out.println(bitTest.getInteger(3));
System.out.println(bitTest.getInteger(2));
}
打印结果:
10
5
12
24
0
0
2.1.3 与、或、非、异或
注意点: and、or、xor方法,都支持多个key之间运算,然后将运算后的结果写入到指定bitmap中。但是,注意的是,bitTest1也会参与运算,也就是bitTest1和bitTest2、bitTest3三者运算后,再将结果写入到bitTest1中,而且,这三者之前,如果key不存在,会按全0处理,所以要考虑key是否存在的情况。
// 与
redissonClient.getBitSet("bitTest1").and("bitTest2","bitTest3");
// 或
redissonClient.getBitSet("bitTest1").or("bitTest2","bitTest3");
// 异或
redissonClient.getBitSet("bitTest1").xor("bitTest2","bitTest3");
// 非
redissonClient.getBitSet("bitTest1").not();