公司做项目的时候,遇到个项目需求:如何在高并发的场景下生成唯一的订单号。
方案一:
如果没有并发,订单号只在一个线程内产生,那么由于程序是顺序执行的,不同订单的生成时间戳正常不同,因此用时间戳+随机数(或自增数)就可以区分各个订单。
如果存在并发,且订单号是由一个进程中的多个线程产生的,那么只要把线程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
优势:简单
劣势:用户不友好,索引关联效率较低