使用redis生成唯一编号及原理示例详解
来源:脚本之家
时间:2022-12-30 12:52:29 284浏览 收藏
在数据库实战开发的过程中,我们经常会遇到一些这样那样的问题,然后要卡好半天,等问题解决了才发现原来一些细节知识点还是没有掌握好。今天golang学习网就整理分享《使用redis生成唯一编号及原理示例详解》,聊聊原理、redis唯一编号,希望可以帮助到正在努力赚钱的你。
在系统开发中,保证数据的唯一性是至关重要的一件事,目前开发中常用的方式有使用数据库的自增序列、UUID生成唯一编号、时间戳或者时间戳+随机数等。
在某些特定业务场景中,可能会要求我们使用特定格式的唯一编号,比如我有一张订单表(t_order),我需要生成“yewu(ORDER)+日期(yyyyMMdd)+序列号(00000000)”格式的订单编号,比如今天的日期是20200716,那我今天第一个订单号就是ORDER2020071600000001、第二个订单号就是ORDER2020071600000002,明天的日期是20200717,那么明天的第一个订单号就是ORDER2020071700000001、第二个订单号就是ORDER2020071700000002,以此类推。
今天介绍下如何使用redis生成唯一的序列号,其实主要思想还是利用redis单线程的特性,可以保证操作的原子性,使读写同一个key时不会出现不同的数据。以SpringBoot项目为例,添加以下依赖。
org.apache.commons commons-lang3 3.1 org.springframework.boot spring-boot-starter-data-redis ${spring.boot.version}
application.properties中配置redis,我本地redis没有设置密码,所以注释了密码这一行
server.port=9091 server.servlet.context-path=/ spring.redis.host=127.0.0.1 spring.redis.port=6379 #spring.redis.password=1234 spring.redis.database=0
创建SequenceService类用于生成特定业务编号
package com.xiaochun.service; import org.apache.commons.lang3.StringUtils; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.support.atomic.RedisAtomicLong; import org.springframework.stereotype.Service; import javax.annotation.Resource; import java.text.SimpleDateFormat; import java.util.Calendar; import java.util.Date; @Service public class SequenceService { private static Logger logger = LoggerFactory.getLogger(SequenceService.class); @Resource private RedisTemplate redisTemplate; //用作存放redis中的key private static String ORDER_KEY = "order_key"; //生成特定的业务编号,prefix为特定的业务代码 public String getOrderNo(String prefix){ return getSeqNo(ORDER_KEY, prefix); } //SequenceService类中公用部分,传入制定的key和prefix private String getSeqNo(String key, String prefix) { Calendar calendar = Calendar.getInstance(); calendar.set(Calendar.HOUR_OF_DAY, 23); calendar.set(Calendar.MINUTE, 59); calendar.set(Calendar.SECOND, 59); calendar.set(Calendar.MILLISECOND, 999); //设置过期时间,这里设置为当天的23:59:59 Date expireDate = calendar.getTime(); //返回当前redis中的key的最大值 Long seq = generate(redisTemplate, key, expireDate); //获取当天的日期,格式为yyyyMMdd String date = new SimpleDateFormat("yyyyMMdd").format(expireDate); //生成八为的序列号,如果seq不够八位,seq前面补0, //如果seq位数超过了八位,那么无需补0直接返回当前的seq String sequence = StringUtils.leftPad(seq.toString(), 8, "0"); if (prefix == null) { prefix = ""; } //拼接业务编号 String seqNo = prefix + date + sequence; logger.info("KEY:{}, 序列号生成:{}, 过期时间:{}", key, seqNo, String.format("%tF %tT ", expireDate, expireDate)); return seqNo; } /** * @param key * @param expireTime 过期时间 * @return */ public static long generate(RedisTemplate redisTemplate,String key,Date expireTime) { //RedisAtomicLong为原子类,根据传入的key和redis链接工厂创建原子类 RedisAtomicLong counter = new RedisAtomicLong(key,redisTemplate.getConnectionFactory()); //设置过期时间 counter.expireAt(expireTime); //返回redis中key的值,内部实现下面详细说明 return counter.incrementAndGet(); } }
接下来,启动项目,使用接口的形式访问,或者写Test方法执行,就可以得到诸如ORDER2020071600000001、ORDER2020071600000002的编号,而且在高并发环境中也不会出现数据重复的情况。实现原理:上面生成特定业务编号主要分为三部分,如下图
前缀和日期部分,没什么需要解释的,主要是redis中的生成的序列号,而这需要依靠RedisAtomicLong来实现,先看下上面生成redis序列过程中发生了什么
- 获取redis中对应业务的key,生成过期时间expireTime
- 获取了RedisTemplate对象,通过该对象获取RedisConnectionFactory对象
- 将key,RedisConnectionFactory对象作为构造参数生成RedisAtomicLong对象,并设置过期时间
- 调用RedisAtomicLong的incrementAndGet()方法
看下RedisAtomicLong源码,当然只放一部分源码,不会放全部,RedisAtomicLong的结构,主要构造函数,和上面提到过的incrementAndGet()方法
public class RedisAtomicLong extends Number implements Serializable, BoundKeyOperations{ private static final long serialVersionUID = 1L; //redis中的key,用volatile修饰,获得原子性 private volatile String key; //当前的key-value对象,根据传入的key获取value值 private ValueOperations operations; //传入当前redisTemplate对象,为RedisTemplate对象的顶级接口 private RedisOperations generalOps; public RedisAtomicLong(String redisCounter, RedisConnectionFactory factory) { this(redisCounter, (RedisConnectionFactory)factory, (Long)null); } private RedisAtomicLong(String redisCounter, RedisConnectionFactory factory, Long initialValue) { Assert.hasText(redisCounter, "a valid counter name is required"); Assert.notNull(factory, "a valid factory is required"); //初始化一个RedisTemplate对象 RedisTemplate redisTemplate = new RedisTemplate(); redisTemplate.setKeySerializer(new StringRedisSerializer()); redisTemplate.setValueSerializer(new GenericToStringSerializer(Long.class)); redisTemplate.setExposeConnection(true); //设置当前的redis连接工厂 redisTemplate.setConnectionFactory(factory); redisTemplate.afterPropertiesSet(); //设置传入的key this.key = redisCounter; //设置当前的redisTemplate this.generalOps = redisTemplate; //获取当前的key-value集合 this.operations = this.generalOps.opsForValue(); //设置默认值,如果传入为null,则key获取operations中的value,如果value为空,设置默认值为0 if (initialValue == null) { if (this.operations.get(redisCounter) == null) { this.set(0L); } //不为空则设置为传入的值 } else { this.set(initialValue); } } //将传入key的value+1并返回 public long incrementAndGet() { return this.operations.increment(this.key, 1L); }
其实主要还是通过redis的自增序列来实现
文中关于redis的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《使用redis生成唯一编号及原理示例详解》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
174 收藏
-
442 收藏
-
439 收藏
-
488 收藏
-
498 收藏
-
342 收藏
-
361 收藏
-
159 收藏
-
164 收藏
-
221 收藏
-
156 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习
-
- 舒服的小懒猪
- 细节满满,码住,感谢作者的这篇技术贴,我会继续支持!
- 2023-03-07 15:30:17
-
- 靓丽的小蘑菇
- 很好,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢大佬分享文章!
- 2023-02-17 04:12:06
-
- 超级的热狗
- 这篇技术文章真及时,细节满满,很棒,已收藏,关注作者了!希望作者能多写数据库相关的文章。
- 2023-02-04 23:36:34
-
- 烂漫的黑夜
- 这篇技术贴真及时,博主加油!
- 2023-01-28 05:25:11
-
- 勤奋的小鸭子
- 细节满满,码起来,感谢up主的这篇博文,我会继续支持!
- 2023-01-21 23:39:34
-
- 落寞的白猫
- 受益颇多,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢up主分享技术贴!
- 2023-01-21 09:37:41
-
- 清秀的信封
- 这篇文章内容真及时,太全面了,真优秀,码起来,关注师傅了!希望师傅能多写数据库相关的文章。
- 2023-01-17 13:19:14
-
- 虚心的香水
- 这篇博文出现的刚刚好,细节满满,受益颇多,mark,关注up主了!希望up主能多写数据库相关的文章。
- 2023-01-12 19:48:01
-
- 温暖的大门
- 这篇博文出现的刚刚好,太详细了,受益颇多,收藏了,关注作者了!希望作者能多写数据库相关的文章。
- 2023-01-07 21:44:40
-
- 英俊的高山
- 真优秀,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢老哥分享博文!
- 2023-01-05 04:19:27
-
- 专注的菠萝
- 这篇博文出现的刚刚好,细节满满,真优秀,已加入收藏夹了,关注楼主了!希望楼主能多写数据库相关的文章。
- 2023-01-03 13:30:08
-
- 幽默的芹菜
- 这篇文章真是及时雨啊,太细致了,写的不错,码起来,关注作者大大了!希望作者大大能多写数据库相关的文章。
- 2023-01-02 23:13:14
-
- 羞涩的期待
- 太给力了,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢up主分享技术贴!
- 2023-01-02 22:27:22
-
- 冷静的冬瓜
- 感谢大佬分享,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢楼主分享技术贴!
- 2022-12-31 14:40:11
-
- 英俊的冷风
- 受益颇多,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢师傅分享技术贴!
- 2022-12-31 13:12:28
-
- 震动的山水
- 这篇技术文章太及时了,老哥加油!
- 2022-12-31 12:21:39