教你如何用Java根據日期生成流水號

前言

生成流水號,在企業中可以說是比較常見的需求,尤其是訂單類業務。

一般來說,需要保證流水號的唯一性。

如果沒有長度和字符的限制,那麼直接使用UUID生成一個唯一字符串即可,也可以直接使用數據庫表中的主鍵,主鍵就是唯一的。

那麼,如果限制瞭流水號必須多少位,這種怎麼生成呢?

可以采用”前綴+日期+數字”的方式(ps:此方式是需要用到緩存的)

前綴:為瞭更好的標識這個流水號是屬於哪種類型;

日期:為瞭防止重復;

數字:為瞭表示當前的流水所處序號。

需求:生成一個17位數的唯一流水號,“LSH”+yyyyMMdd+6位數字

代碼實現

import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.Date;
import java.util.GregorianCalendar;
import java.util.concurrent.atomic.AtomicInteger;

public class SerialNoTest {

    public static void main(String[] args) {
        String serialNo = generateSerialNo();
        System.out.println("生成的流水號:"+serialNo);
    }

    /**
     * 生成17位唯一流水號,"LSH"+yyyyMMdd+6位數字
     * 6位數字,如:000001
     * @return
     */
    private static String generateSerialNo(){
        //定義需要返回的流水號
        String serialNo = null;
        //先查詢到今天的日期,格式:"yyyyMMdd"
        String todayDate = new SimpleDateFormat("yyyyMMdd")
                                .format(new Date());
        //固定字母前綴 拼接 今天日期,組成新的完整的前綴,也就是緩存的key
        String cacheKey = "LSH"+todayDate;
        //再通過key查詢緩存有沒有num數據,緩存操作根據自身項目封裝工具類
        Long codeNum = cacheService.getCache(cacheKey, Long.class);
        //如果緩存查詢有值,數值+1,再賦值給下一個流水號
        if (null != codeNum) {
            codeNum = codeNum + 1L;
        } else {
            //如果緩存查詢沒值,直接賦值為1
            codeNum = 1L;
        }
        //流水號 = 緩存key + 拼接的數值 = 前綴 + 日期 + 拼接的數值
        serialNo = getCodeOfSix(cacheKey, codeNum.intValue());
        //設置緩存,調用此方法,會自動將key所對應的value+1,保存時長:今天剩餘的時間
        cacheService.incr(cacheKey, getSeconds());
        return serialNo;
    }


    /**
     * 將數值拼接成對應的位數
     * @param prefix  前綴:"LSH"+yyyyMMdd
     * @param nowNum  當前要生成的數字
     * @return 拼接好的流水號
     */
    public static String getCodeOfSix(String prefix,int nowNum ) {
        //需要返回的code
        StringBuilder codeSb = new StringBuilder();
        //需要拼接的數字
        StringBuilder numSb = new StringBuilder();
        //封裝的數字對象,裡面 value 加瞭 volatile關鍵字,保證瞭線程安全
        AtomicInteger count = new AtomicInteger(nowNum);

        //將數值補足為6位字符串
        if (count.get() < 10) {
            numSb.append("00000").append(count.get());
        } else if(count.get() < 100){
            numSb.append("0000").append(count.get());
        }else if(count.get() < 1000){
            numSb.append("000").append(count.get());
        }else if(count.get() < 10000){
            numSb.append("00").append(count.get());
        }else if(count.get() < 100000){
            numSb.append("0").append(count.get());
        } else if (count.get() >= 100000) {
            numSb.append(count.get());
        }

        //先拼接前綴
        codeSb.append(prefix);
        //再拼接數字
        codeSb.append(numSb);
        return codeSb.toString();
    }


    /**
     * 獲取當天結束還剩餘多少秒
     * @return
     */
    public static int getSeconds(){
		//獲取今天當前時間
        Calendar curDate = Calendar.getInstance();
		//獲取明天凌晨0點的日期
        Calendar tommorowDate = new GregorianCalendar(
                curDate.get(Calendar.YEAR),
                curDate.get(Calendar.MONTH), 
                curDate.get(Calendar.DATE) + 1,
                0, 0, 0);
		//返回 明天凌晨0點 和 今天當前時間 的差值(秒數) 
        return (int)(tommorowDate.getTimeInMillis() - curDate .getTimeInMillis()) / 1000;
    }
}

假如今天是2021年4月22日,運行項目,生成的第1個流水號則為:LSH20210422000001

第2個流水號則為:LSH20210422000002,依次類推。

需要註意的是:

如果限制瞭位數,6位數字每天最多能生成10w個流水號,所以,這個數字位數根據具體業務量進行調整。

如果每天的生成數量量不到1w,那麼使用4位數字即可。

到此這篇關於教你如何用Java根據日期生成流水號的文章就介紹到這瞭,更多相關java生成流水號內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: