高并发下如何生成唯一订单号

公司做项目的时候,遇到个项目需求:如何在高并发的场景下生成唯一的订单号。

方案一:

如果没有并发,订单号只在一个线程内产生,那么由于程序是顺序执行的,不同订单的生成时间戳正常不同,因此用时间戳+随机数(或自增数)就可以区分各个订单。
如果存在并发,且订单号是由一个进程中的多个线程产生的,那么只要把线程ID添加到序列号中就可以保证订单号唯一。
如果存在并发,且订单号是由同一台主机中的多个进程产生的,那么只要把进程ID添加到序列号中就可以保证订单号唯一。
如果存在并发,且订单号是由不同台主机产生的,那么MAC地址、IP地址或CPU序列号等能够区分主机的号码添加到序列号中就可以保证订单号唯一。

方案二:
时间戳+用户ID+几个随机数+乐观锁。

方案三:
用redis的原子递增,做好高可用集群。

方案四(非纯数字):
java自带uuid。

事例代码

java获取UUID

UUID.randomUUID().toString()

java获取线程ID

Thread.currentThread().getId()

java获取进程ID

//java获取进程ID
RuntimeMXBean runtime = ManagementFactory.getRuntimeMXBean();
Field jvm = runtime.getClass().getDeclaredField("jvm");
jvm.setAccessible(true);
VMManagement mgmt = (VMManagement) jvm.get(runtime);
Method pidMethod = mgmt.getClass().getDeclaredMethod("getProcessId");
pidMethod.setAccessible(true);
int pid = (Integer) pidMethod.invoke(mgmt);

java获取mac地址

InetAddress ia = InetAddress.getLocalHost();
byte[] mac = NetworkInterface.getByInetAddress(ia).getHardwareAddress();
String macStr = DatatypeConverter.printHexBinary(mac);

那么他们各自的优缺点又是什么呢?

1.数据库自增长ID
优势:无需编码
缺陷:
大表不能做水平分表,否则插入删除时容易出现问题。
高并发下插入数据需要加入事务机制。
在业务操作父、子表(关联表)插入时,先要插入父表,再插入子表。

2.时间戳+随机数
优势:编码简单
缺陷:随机数存在重复问题,即使在相同的时间戳下。每次插入数据库前需要校验下是否存在相同的数值。

3.时间戳+会员ID
优势:同一时间,一个用户不会存在两张订单。
缺陷: 会员ID也会透露运营数据,鸡生蛋、蛋生鸡的问题。

4.GUID/UUID
优势:简单
劣势:用户不友好,索引关联效率较低

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值