基于Redis生成全局ID

本文介绍如何使用Java配合Redis高效生成有序的订单号,包括两种方法:时间戳加自增和Redis原子操作,并探讨其优缺点,以及在Redis集群中的应用实例。

【Java毕设合集】20套毕设系统项目(附源码),任意挑选,进来白嫖_20个Java练手项目_Java开发_Java项目_哔哩哔哩_bilibili

方法一:预先生成

1、提前生成一批订单号放在redis中,每次使用的时候取一个,当库里没有了,再生成一批
2、使用的时候前面用系统时间戳,后面加上redis中取得的数据即可。

方法二:利用redis的自增

因为Redis是单线的,天生保证原子性,可以使用Redis的原子操作 INCR和INCRBY来实现

优点

不依赖于数据库,灵活方便,且性能优于数据库。
数字ID天然排序,对分页或者需要排序的结果很有帮助。

缺点

如果系统中没有Redis,还需要引入新的组件,增加系统复杂度。
需要编码和配置的工作量比较大。

注意:在Redis集群情况下,同样和Redis一样需要设置不同的增长步长,同时key一定要设置有效期

》可以使用Redis集群来获取更高的吞吐量。假如一个集群中有5台Redis。可以初始化每台Redis的值分别是1,2,3,4,5,然后步长都是5。各个Redis生成的ID为:

A:1,6,11,16,21
B:2,7,12,17,22
C:3,8,13,18,23
D:4,9,14,19,24
E:5,10,15,20,25

比较适合使用Redis来生成每天从0开始的流水号。比如订单号=日期+当日自增长号。可以每天在Redis中生成一个Key,使用INCR进行累加。
如果生成的订单号超过自增增长的话,可以采用前缀+自增+并且设置有效期

设置有效期可统计当天的数据量,

eg:2019010409552100001、2019010409552100002、2019010409552200003、2019010409552300004
2019010509552100001、2019010509552100002、2019010509552200003、2019010509552300004

代码示例:

@RestController
public class OrderController {
    @Autowired
    private RedisTemplate redisTemplate;

    /**
    * timeOut一般设置为24小时,则次日会从0重新开始计算
    */
    @RequestMapping("/order")
    public String order(String key, Long timeOut) {
        RedisAtomicLong redisAtomicLong = new RedisAtomicLong(key, redisTemplate.getConnectionFactory());
        //设置起始值
        // redisAtomicLong.set(1);
        // 设置redis步长增长为2
        redisAtomicLong.addAndGet(1);
        // for (int i = 0; i < 100; i++) {
        long andIncrement = redisAtomicLong.getAndIncrement();
        //设置5位,不够则前面补零
        String orderId = prefix() + String.format("%1$05d", andIncrement);
        String insertSQL = "insert into orderNumber value('" + orderId + "');";
        // System.out.println(Thread.currentThread().getName() +
        // ",insertSQL:" + insertSQL);
        System.out.println(insertSQL);
        if ((null == redisAtomicLong || redisAtomicLong.longValue() == 0) && timeOut > 0) {// 初始设置过期时间
            redisAtomicLong.expire(timeOut, TimeUnit.SECONDS);
        }

        // }

        return "success";
    }

    public static String prefix() {
        String temp_str = "";
        Date dt = new Date();
        // 最后的aa表示“上午”或“下午” HH表示24小时制 如果换成hh表示12小时制
        SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmss");
        temp_str = sdf.format(dt);
        return temp_str;
    }
    public static void main(String[] args) {
        System.out.println(prefix());
    }
}


 

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值