我的编程空间,编程开发者的网络收藏夹
学习永远不晚

怎么优雅地使用Redis位图操作

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

怎么优雅地使用Redis位图操作

本篇内容主要讲解“怎么优雅地使用Redis位图操作”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么优雅地使用Redis位图操作”吧!

在进入今天的主题前,先简单地解释下Redis中的位图到底是什么。Redis官方文档对于位图的介绍如下:

位图不是一个真实的数据类型,而是定义在字符串类型上的面向位的操作的集合。由于字符串类型是二进制安全的二进制大对象,并且最大长度是 512MB,适合于设置 2^32个不同的位。

位操作分为两组:常量时间单个位的操作,像设置一个位为 1 或者 0,或者获取该位的值。对一组位的操作,例如计算指定范围位的置位数量。

位图的最大优势是有时是一种非常显著的节省空间来存储信息的方式。例如,在一个系统中,不同用户由递增的用户 ID 来表示,可以使用 512MB 的内存来表示 400 万用户的单个位信息(例如他们是否需要接收信件)。

简而言之,位图操作是用来操作比特位的,其优点是节省内存空间。为什么可以节省内存空间呢?假如我们需要存储100万个用户的登录状态,使用位图的话最少只需要100万个比特位(比特位1表示登录,比特位0表示未登录)就可以存储了,而如果以字符串的形式存储,比如说以userId为key,是否登录(字符串“1”表示登录,字符串“0”表示未登录)为value进行存储的话,就需要存储100万个字符串了,相比之下使用位图存储占用的空间要小得多,这就是位图存储的优势。

位图常用操作

位图的常用操作如下:

·setbit

设置特定key对应的比特位的值。

·getbit

获取特定key对应的比特位的值。

·bitcount

统计给定key对应的字符串比特位为1的数量。

使用位图存储用户登录状态

位图的常见应用是用来存储状态值,比如存储用户的登录状态。

假设我们现在有一个需求,需要记录用户注册以来每天的登录状态,那么我们就可以以用户id为key,然后以日期或者日期的偏移量作为下标,将登录状态存储到对应的比特位中,这样就可以很方便地获取用户某一天的登录状态了。

接下来看代码:

public class UserLoginStatusService {      private static final String host="111.111.111.111";      private static final int port=6379;      private static final Jedis jedis=new Jedis(host,port);      //日期的初始值(也可以理解为用户的注册时间),     //下文需要使用日期的偏移量作为redis位图的offset,     //因此需要将要保存登录状态的日期减去该初始日期。     //这里使用了Java 8的新日期API     private static final LocalDate beginDate=LocalDate.of(2018,1,1);      static {         jedis.connect();     }      public void setLoginStatus(String userId, LocalDate date,boolean isLogin){         long offset = getDateDuration(beginDate, date);         jedis.setbit(userId,offset,isLogin);     }      public boolean getLoginStatus(String userId,LocalDate date){         long offset = getDateDuration(beginDate, date);         return jedis.getbit(userId,offset);     }      private long getDateDuration(LocalDate start ,LocalDate end){         return start.until(end, ChronoUnit.DAYS);     }      public static void main(String[] args) {         UserLoginStatusService userLoginStatusService=new UserLoginStatusService();         String userId="user_1";         LocalDate today = LocalDate.now();         userLoginStatusService.setLoginStatus(userId,today,true);         boolean todayLoginStatus = userLoginStatusService.getLoginStatus(userId, today);         System.out.println(String.format("The loginStatus of %s in %s is %s",userId,today,todayLoginStatus));         LocalDate yesterday = LocalDate.now().minusDays(1);         boolean yesterdayLoginStatus = userLoginStatusService.getLoginStatus(userId, yesterday);         System.out.println(String.format("The loginStatus of %s in %s is %s",userId,yesterday,yesterdayLoginStatus));     }  }

代码不复杂,我们在main方法中设置当天的登录状态为true,然后分别查出当天的登录状态和昨天的登录状态,由于redis位图的比特位默认是0,所以该代码的正确输出应该是今天已登录,昨天未登录,我们运行一次看看结果。

怎么优雅地使用Redis位图操作

从程序运行结果来看,Redis的位图确实满足了我们的需求,且兼有节省存储空间的优点。

使用位图统计登录天数

接下来我们有一个新需求,就是统计某个用户注册后前10天的登录天数,Redis中有个bitcount命令,可以统计某个字符串中的比特位为1的数量,其还有2个参数start和end,表示要统计的范围,咋一看好像可以用来实现我们这个需求,但是这里有一个坑需要注意下,bitcount命令的start和end参数指的是字节的索引,而不是比特位的索引,而我们如果要使用位图来统计某个用户注册后前10天的登录天数的话,需要统计的是比特位索引从0到9的比特值为1的数量,所以直接使用bitcount命令显然是无法满足要求的。那么假如说我们一定要用位图来存储登录状态呢,应该咋办呢?其实办法还是有的。我们可以先拿到比特位索引从0到9所在的字节数组,再将该字节数组解析成二进制形式,进而统计出比特位索引从0到9比特值为1的数量。

要拿到比特位索引所在的字节在字节数组中的下标比较简单,只要将比特位索引除以8(一个字节包含8个比特位)再向下取整就行了。接下来就是使用redis的getrange命令来截取字节数组了。

拿到了字节数组,接下来就是解析字节数组,统计其中比特值为1的数量了。我们先从最简单的单个字节说起,假设一个字节的各个比特位的值如下:

怎么优雅地使用Redis位图操作

我们设比特位索引为index,假如我们要计算比特位为7的比特值,只需要将原值直接跟1进行与运算就行了。要计算比特位为6的比特值,只需要将原值右移1位,再跟1进行与运算。以此类推,要计算第index位的比特值,只需要先右移(7-index)位,再跟1进行与运算即可。

只要能够统计出截取出来的的字节数组中比特位的值为1的数量,接下来再减去不包含在对应比特索引中的比特值为1的数量,即可统计出给定的比特索引范围内比特值为1的数量。

这么说有点拗口,我们以上述例子为例进行讲解吧。我们要统计出用户注册后前10天的登录天数,如果用位图存储用户登录状态,位图中的索引为注册天数的话,那么我们需要统计比特索引从0到9的比特值为1的数量,才能计算出该用户注册后前10天的登录天数。

我们先计算出比特索引从0到9包含在哪一段字节数组中,前面说了,只需要将对应的索引除以8,再向下取整就行了。从而可以得知比特位索引从0到9对应的是下标从0到1的字节数组。

接下来使用getrange命令截取该字节数组,假设其值如下:

怎么优雅地使用Redis位图操作

怎么优雅地使用Redis位图操作

假设比特索引0到9对应的字节数组的比特值情况如上所示,我们需要统计的是第一个字节(下标为0)中的0到7位中比特值为1的数量,再加上第二个字节(下标为1)中的第0到1位中比特值为1的数量。加起来刚好10位,也就是对应用户注册前10天的登录天数。当然我们也可以统计出这2个字节中的比特值为1的总数,再减去第二个字节的从2到7位(上述表格标红的地方)中比特值为1的数量,也可统计出该用户注册后前10天的登录天数。本文用的是第二种方法。

接下来上代码:

private static final int BIT_AMOUNT_IN_ONE_BYTE =8;      private Jedis jedis;       public int bitCountByBitIndex(String key, long startBitIndex, long endBitIndex) {         int startByteIndex = getByteIndexInTheBytes(startBitIndex);         int endByteIndex = getByteIndexInTheBytes(endBitIndex);         byte[] bytes = jedis.getrange(key.getBytes(), startByteIndex, endByteIndex);         int totalBitInBytes = getTotalBitInBytes(bytes);         int startBitIndexInFirstByte = getBitIndexInTheByte(startBitIndex);         int endBitIndexInLastByte = getBitIndexInTheByte(endBitIndex);         byte firstByte = bytes[0];         byte lastByte = bytes[bytes.length-1];         for(int i=7;i>(BIT_AMOUNT_IN_ONE_BYTE-1-startBitIndexInFirstByte);i--){             if(((firstByte>>i)&1)==1){                 totalBitInBytes--;             }         }         for(int i=0;i<(BIT_AMOUNT_IN_ONE_BYTE-1-endBitIndexInLastByte);i++){             if(((lastByte>>i)&1)==1){                 totalBitInBytes--;             }         }          return totalBitInBytes;     }      private int getTotalBitInBytes(byte[] bytes){         int count=0;         for(byte b:bytes){             for(int i = 0; i< BIT_AMOUNT_IN_ONE_BYTE; i++){                 if(((b>>i)&1)==1){                     count++;                 }             }         }         return count;     }      private int getByteIndexInTheBytes(long offset){         return (int) offset/ BIT_AMOUNT_IN_ONE_BYTE;     }      private int getBitIndexInTheByte(long offset){         return (int)(offset-offset/ BIT_AMOUNT_IN_ONE_BYTE * BIT_AMOUNT_IN_ONE_BYTE);     }

代码就不注释了,整体思路已经在上面讲解了。

当然要实现本文所述的功能,也不一定非要这么做,还是有其他的方案的。比如:可以将放入位图的offset统一乘以8(一个字节占8比特),这样一来就可以直接用redis的bitcount来统计对应索引范围内的比特值为1的数量了,当然这种方案的缺点也相当明显,就是浪费内存,因为原先只需要1比特存储的数据,现在需要8比特存储,所以这种方案不能很好地利用位图索引节省存储空间的特点。

到此,相信大家对“怎么优雅地使用Redis位图操作”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

怎么优雅地使用Redis位图操作

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

怎么优雅地使用Redis位图操作

本篇内容主要讲解“怎么优雅地使用Redis位图操作”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么优雅地使用Redis位图操作”吧!在进入今天的主题前,先简单地解释下Redis中的位图到底是什
2023-06-05

怎么使用PHP操作Redis

本篇内容主要讲解“怎么使用PHP操作Redis”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么使用PHP操作Redis”吧!redis 的基本操作方法1 redis 的连接 ://实例化red
2023-06-20

怎么使用Python操作Redis数据库

本篇内容主要讲解“怎么使用Python操作Redis数据库”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么使用Python操作Redis数据库”吧!介绍Redis是一个开源的基于内存也可持久化
2023-07-02

IDEA版怎么使用Java操作Redis数据库

本篇内容主要讲解“IDEA版怎么使用Java操作Redis数据库”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“IDEA版怎么使用Java操作Redis数据库”吧!首先 下载 jedis.jar包
2023-06-20

使用python怎么批量操作redis数据库

使用python怎么批量操作redis数据库?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。方法一:使用 pipeline  使用pipelining 发送命令时
2023-06-08

SpringBoot怎么使用RedisTemplate操作Redis数据类型

这篇文章主要讲解了“SpringBoot怎么使用RedisTemplate操作Redis数据类型”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“SpringBoot怎么使用RedisTempl
2023-06-29

Java怎么使用责任链默认优雅地进行参数校验

本篇内容介绍了“Java怎么使用责任链默认优雅地进行参数校验”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!前言项目中参数校验十分重要,它可以
2023-07-05

怎么使用Python操作xmind绘制思维导图

这篇文章主要介绍“怎么使用Python操作xmind绘制思维导图”,在日常操作中,相信很多人在怎么使用Python操作xmind绘制思维导图问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么使用Python操
2023-06-25

vue使用高德地图根据坐标定位点的代码怎么写

今天小编给大家分享一下vue使用高德地图根据坐标定位点的代码怎么写的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。代码如下
2023-07-04

Win7 64位操作系统怎么使用DVD刻录光驱复制软件

如果你有一台电脑配备了DVD刻录javascriptRxcOAnbMZz光驱,那稍微有点电脑常识的朋友就一定知道可以用它来进行刻录自己想要的光盘。那么,如何复制DVD光盘就成了一个很现实的问题了。如果你是一个菜鸟,你把DVD光盘放入光驱,你
2023-06-12

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录