記一次訂單號的重復(fù)事故

系統(tǒng)出現(xiàn)了兩個一模一樣的訂單號,訂單的內(nèi)容卻不是不一樣的,而且系統(tǒng)在按照 訂單號查詢的時候一直拋錯,也沒法正常回調(diào),而且事情發(fā)生的不止一次,所以 這次系統(tǒng)升級一定要解決掉。
??????/**
????*?OD單號生成
????*?訂單號生成規(guī)則:OD + yyMMddHHmmssSSS + 5位數(shù)(商戶ID3位+隨機數(shù)2位) 22位
????*/
???public?static?String?getYYMMDDHHNumber(String?merchId){
??????????StringBuffer?orderNo?=?new?StringBuffer(new?SimpleDateFormat("yyMMddHHmmssSSS").format(new?Date()));
??????????if(StringUtils.isNotBlank(merchId)){
??????????????if(merchId.length()>3){
??????????????????orderNo.append(merchId.substring(0,3));
??????????????}else?{
??????????????????orderNo.append(merchId);
??????????????}
??????????}
??????????int?orderLength?=?orderNo.toString().length();
??????????String?randomNum?=?getRandomByLength(20-orderLength);
??????????orderNo.append(randomNum);
??????????return?orderNo.toString();
???}
??
??
??????/**?生成指定位數(shù)的隨機數(shù)?**/
??????public?static?String?getRandomByLength(int?size){
??????????if(size>8?||?size<1){
??????????????return?"";
??????????}
??????????Random?ne?=?new?Random();
??????????StringBuffer?endNumStr?=?new?StringBuffer("1");
??????????StringBuffer?staNumStr?=?new?StringBuffer("9");
??????????for(int?i=1;i??????????????endNumStr.append("0");
??????????????staNumStr.append("0");
??????????}
??????????int?randomNum?=?ne.nextInt(Integer.valueOf(staNumStr.toString()))+Integer.valueOf(endNumStr.toString());
??????????return?String.valueOf(randomNum);
??????}
????public?static?void?main(String[]?args)?{
????????final?String?merchId?=?"12334";
????????List?orderNos?=?Collections.synchronizedList(new?ArrayList ());
????????IntStream.range(0,100).parallel().forEach(i->{
????????????orderNos.add(getYYMMDDHHNumber(merchId));
????????});
????????List?filterOrderNos?=?orderNos.stream().distinct().collect(Collectors.toList());
????????System.out.println("生成訂單數(shù):"+orderNos.size());
????????System.out.println("過濾重復(fù)后訂單數(shù):"+filterOrderNos.size());
????????System.out.println("重復(fù)訂單數(shù):"+(orderNos.size()-filterOrderNos.size()));
????}
生成訂單數(shù):100
過濾重復(fù)后訂單數(shù):87
重復(fù)訂單數(shù):13
去掉商戶ID的傳入(按同事的說法,傳入商戶ID也是為了防止重復(fù)訂單的,事實證明并沒有叼用) 毫秒僅保留三位(縮減長度同時保證應(yīng)用切換不存在重復(fù)的可能) 使用線程安全的計數(shù)器做數(shù)字遞增(三位數(shù)最低保證并發(fā)800不重復(fù),代碼中我給了4位) 更換日期轉(zhuǎn)換為java8的日期類以格式化(線程安全及代碼簡潔性考量)
????/**?訂單號生成(NEW)?**/
????private?static?final?AtomicInteger?SEQ?=?new?AtomicInteger(1000);
????private?static?final?DateTimeFormatter?DF_FMT_PREFIX?=?DateTimeFormatter.ofPattern("yyMMddHHmmssSS");
????private?static?ZoneId?ZONE_ID?=?ZoneId.of("Asia/Shanghai");
????public?static?String?generateOrderNo(){
????????LocalDateTime?dataTime?=?LocalDateTime.now(ZONE_ID);
????????if(SEQ.intValue()>9990){
????????????SEQ.getAndSet(1000);
????????}
????????return??dataTime.format(DF_FMT_PREFIX)+SEQ.getAndIncrement();
????}
????public?static?void?main(String[]?args)?{
????????List?orderNos?=?Collections.synchronizedList(new?ArrayList ());
????????IntStream.range(0,8000).parallel().forEach(i->{
????????????orderNos.add(generateOrderNo());
????????});
????????List?filterOrderNos?=?orderNos.stream().distinct().collect(Collectors.toList());
????????System.out.println("生成訂單數(shù):"+orderNos.size());
????????System.out.println("過濾重復(fù)后訂單數(shù):"+filterOrderNos.size());
????????System.out.println("重復(fù)訂單數(shù):"+(orderNos.size()-filterOrderNos.size()));
????}
????
????/**
????????測試結(jié)果:?
????????生成訂單數(shù):8000
????????過濾重復(fù)后訂單數(shù):8000
????????重復(fù)訂單數(shù):0
????**/
使用UUID(在第一次生成訂單號時初始化一個)
使用redis記錄一個增長ID
使用數(shù)據(jù)庫表維護一個增長ID
應(yīng)用所在的網(wǎng)絡(luò)IP
應(yīng)用所在的端口號
使用第三方算法(雪花算法等等)
使用進程ID(某種程度下是一個可行的方案)
import?org.apache.commons.lang3.RandomUtils;
import?java.net.InetAddress;
import?java.time.LocalDateTime;
import?java.time.ZoneId;
import?java.time.format.DateTimeFormatter;
import?java.util.ArrayList;
import?java.util.Collections;
import?java.util.List;
import?java.util.concurrent.atomic.AtomicInteger;
import?java.util.stream.Collectors;
import?java.util.stream.IntStream;
public?class?OrderGen2Test?{
????/**?訂單號生成?**/
????private?static?ZoneId?ZONE_ID?=?ZoneId.of("Asia/Shanghai");
????private?static?final?AtomicInteger?SEQ?=?new?AtomicInteger(1000);
????private?static?final?DateTimeFormatter?DF_FMT_PREFIX?=?DateTimeFormatter.ofPattern("yyMMddHHmmssSS");
????public?static?String?generateOrderNo(){
????????LocalDateTime?dataTime?=?LocalDateTime.now(ZONE_ID);
????????if(SEQ.intValue()>9990){
????????????SEQ.getAndSet(1000);
????????}
????????return??dataTime.format(DF_FMT_PREFIX)+?getLocalIpSuffix()+SEQ.getAndIncrement();
????}
????private?volatile?static?String?IP_SUFFIX?=?null;
????private?static?String?getLocalIpSuffix?(){
????????if(null?!=?IP_SUFFIX){
????????????return?IP_SUFFIX;
????????}
????????try?{
????????????synchronized?(OrderGen2Test.class){
????????????????if(null?!=?IP_SUFFIX){
????????????????????return?IP_SUFFIX;
????????????????}
????????????????InetAddress?addr?=?InetAddress.getLocalHost();
????????????????//??172.17.0.4??172.17.0.199?,
????????????????String?hostAddress?=?addr.getHostAddress();
????????????????if?(null?!=?hostAddress?&&?hostAddress.length()?>?4)?{
????????????????????String?ipSuffix?=?hostAddress.trim().split("\\.")[3];
????????????????????if?(ipSuffix.length()?==?2)?{
????????????????????????IP_SUFFIX?=?ipSuffix;
????????????????????????return?IP_SUFFIX;
????????????????????}
????????????????????ipSuffix?=?"0"?+?ipSuffix;
????????????????????IP_SUFFIX?=?ipSuffix.substring(ipSuffix.length()?-?2);
????????????????????return?IP_SUFFIX;
????????????????}
????????????????IP_SUFFIX?=?RandomUtils.nextInt(10,?20)?+?"";
????????????????return?IP_SUFFIX;
????????????}
????????}catch?(Exception?e){
????????????System.out.println("獲取IP失敗:"+e.getMessage());
????????????IP_SUFFIX?=??RandomUtils.nextInt(10,20)+"";
????????????return?IP_SUFFIX;
????????}
????}
????public?static?void?main(String[]?args)?{
????????List?orderNos?=?Collections.synchronizedList(new?ArrayList ());
????????IntStream.range(0,8000).parallel().forEach(i->{
????????????orderNos.add(generateOrderNo());
????????});
????????List?filterOrderNos?=?orderNos.stream().distinct().collect(Collectors.toList());
????????System.out.println("訂單樣例:"+?orderNos.get(22));
????????System.out.println("生成訂單數(shù):"+orderNos.size());
????????System.out.println("過濾重復(fù)后訂單數(shù):"+filterOrderNos.size());
????????System.out.println("重復(fù)訂單數(shù):"+(orderNos.size()-filterOrderNos.size()));
????}
}
/**
??訂單樣例:20082115575546011022
??生成訂單數(shù):8000
??過濾重復(fù)后訂單數(shù):8000
??重復(fù)訂單數(shù):0
**/
最后
generateOrderNo()方法內(nèi)不需要加鎖,因為AtomicInteger內(nèi)使用的是CAS自旋轉(zhuǎn)鎖(保證可見性的同時也保證原子性,具體的請自行了解) getLocalIpSuffix()方法內(nèi)不需要對不為null的邏輯加同步鎖(雙向校驗鎖,整體是一種安全的單例模式) 本人實現(xiàn)的方式并不是解決問題的唯一方式,具體解決問題需要視當(dāng)前系統(tǒng)架構(gòu)具體而論 任何測試都是必要的,我同事在前幾次嘗試解決這個問題后都沒有自測,不測試有損開發(fā)專業(yè)性!
往期推薦


評論
圖片
表情
