不依赖外部流水号,完全靠时间戳和随机数生成订单号无法避免冲突,所以必须引入外部的流水号生成机制。或使用数据库,或使用APC之类的缓存。用APC之类的缓存存在一个问题,就是无法持久保持数据,服务器重启或者PHP宿主进程重启都会清空流水号计数器,所以可以采取缓存+数据库结合的模式——如果缓存中有流水号计数器数据则读取并累加计数,如果缓存中没有流水号计数器从数据库中还原计数器。计数器可以每隔一段时间重置一次。
既然引入了自增流水号计数器,又会导致文章开头的“德国坦克问题”,所以需要用skip32算法把流水号加密(https://github.com/nlenepveu/Skip32)。
基于前文所述,得到如下的订单号生成规则:
订单号 = 日期前缀 + 加密流水号
// Skip32 算法加密密钥
const ENCRYPTED_KEY = ‘xxxxxxxxxxxx';
// 使用 Wincache 作为流水号计数器缓存
function getOrderSerialNumber() {
$timestamp = time();
$datePrefix = date(‘ymd’, $timestamp);
// 如果流水号计数器数据不在缓存中,则尝试从数据库中恢复
if (false === ($value = wincache_ucache_inc($datePrefix))) {
wincache_lock($datePrefix);
// 从数据库中获取今日的订单数
$counter = getNumberOfOrdersTodayFromDatabase($timestamp);
$value = $counter + 1;
if (!wincache_ucache_add($datePrefix, $value, 60*60*24)) {
$value = wincache_ucache_inc($datePrefix);
}
wincache_unlock($datePrefix);
}
return $datePrefix.str_pad(Skip32::encrypt($datePrefix.ENCRYPTED_KEY, $value), 10, ‘0’, STR_PAD_LEFT);
}