支付不重复订单号生成

14 篇文章 0 订阅
1 篇文章 0 订阅

最近的项目中支付的订单号是使用附加项目中的工具做的,是一种带数据库自增形式的订单号,开始测试的时候没有任何问题,但是再高并发的实际应用中还是有不小的问题,只怪自己考虑太少,必须在高并发模式下保证订单号的唯一,我也参考了很多的案例,网上有个网友的回复很是精辟:

如果没有并发,订单号只在一个线程内产生,那么由于程序是顺序执行的,不同订单的生成时间一定不同,因此用时间就可以区分各个订单。

如果存在并发,且订单号是由一个进程中的多个线程产生的,那么只要把线程ID添加到序列号中就可以保证订单号唯一。

如果存在并发,且订单号是由同一台主机中的多个进程产生的,那么只要把进程ID添加到序列号中就可以保证订单号唯一。

如果存在并发,且订单号是由不同台主机产生的,那么MAC地址、IP地址或CPU序列号等能够区分主机的号码添加到序列号中就可以保证订单号唯一。

地址:https://www.oschina.net/question/1757031_2137794

还有种方法是再订单号中加入时间,用户id(或唯一标识),随机数:其实这种方法对于那种业务并发低一些的项目中,这不失为一种好并发,像我最近做的停车项目中用户就不会再同一时间内停多辆车,这不失为一种办法

我做的是

    /**
	 * 获得唯一订单号
	 */
	public static String getUniqueOrder() {
		 SimpleDateFormat format = new SimpleDateFormat("YYYYMMddHHmmss");
		 String format2 = format.format(new Date());
		 int hashCodeV = UUID.randomUUID().toString().hashCode();  
		 if(hashCodeV < 0) {
			//有可能是负数
			 hashCodeV = - hashCodeV;  
		 }
		 return "pk"+format2+String.format("%012d", hashCodeV);  
	}

以后做到多做笔记,记录下项目中遇到的点点滴滴

为了实现这个需求,你可以在不依赖数据库唯一索引的情况下,采用一些策略确保订单号唯一性: 1. **业务层面校验**:首先,在接收调用方传来的订单号和金额之前,可以在服务端进行简单的验证。检查当前系统内的历史记录或者缓存中是否存在相同的订单号,如果存在则返回错误信息。 2. **UUID生成**:使用全局唯一的ID生成算法,比如Java的`UUID`,这样每次生成订单号都是独一无二的,降低冲突的可能性。 3. **自增序列或编号生成**:虽然数据库中无法直接创建唯一索引,但在插入新订单时,你可以在程序中维护一个递增的订单ID,然后将这个ID与生成唯一标识(如UUID的一部分)结合,形成最终的订单号。这样既保持了顺序,又提高了唯一性。 4. **分布式锁定**:对于高并发情况,可能需要使用分布式锁技术,如Redis或Zookeeper,确保在创建订单过程中只有一个请求能够成功。请求到达后,尝试获取锁,成功获取后再操作数据库,失败则等待。 5. **幂等性处理**:确保对同一条订单号的多次调用能够达到同样的效果,比如在订单表中添加一个版本字段,只有当版本号增加时才表示新的订单创建,旧的版本号则视为重复。 下面是简单示例代片段,假设我们使用了UUID和一个自增长的订单ID: ```java import java.util.UUID; public class OrderService { private static Map<String, Integer> orderMap = new ConcurrentHashMap<>(); public String createOrder(String orderId, double amount) throws OrderExistsException { if (orderMap.containsKey(orderId)) { throw new OrderExistsException("Order already exists with ID " + orderId); } int version = getOrderVersion(orderId); // 结合订单号和版本生成最终的订单ID String finalOrderId = orderId + "_" + version; // 更新订单状态并在数据库保存 saveToDatabase(finalOrderId, amount); // 更新映射和版本号 orderMap.put(finalOrderId, version + 1); return finalOrderId; } private void saveToDatabase(String orderId, double amount) { // 实际数据库操作,这里省略 } private int getOrderVersion(String orderId) { // 获取订单ID对应的版本,这里也可以从缓存或数据库查询,实际实现根据实际情况 // 示例代只返回初始值1 return 1; } // ...其他错误处理和事务管理逻辑 } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值