-----2017-04-17更新-----
- Re: claram 2017-02-04 16:15发表 [回复]
-
- 回复qq_27482239:line 36: 改为 >=
- line 41:finOderNum 改为 finOrderNum
感谢
claram 指正
PS:本方法可能较为笨重,应用中高并发存在重复的可能性,可参考下一篇文章《Java订单号生成工具(实现二)基于队列》
-----2017-04-17更新-----
Java实现的订单号生成工具,生成非重复订单号,可设置每毫秒生成的订单最大值。
用到了对象锁。
- import java.text.SimpleDateFormat;
- import java.util.Date;
-
- /**
- * @ClassName: MakeOrderNum
- * @CreateTime 2015年9月13日 下午4:51:02
- * @author : mayi
- * @Description: 订单号生成工具,生成非重复订单号,理论上限1毫秒1000个,可扩展
- *
- */
- public class MakeOrderNum {
- /**
- * 锁对象,可以为任意对象
- */
- private static Object lockObj = "lockerOrder";
- /**
- * 订单号生成计数器
- */
- private static long orderNumCount = 0L;
- /**
- * 每毫秒生成订单号数量最大值
- */
- private int maxPerMSECSize=1000;
- /**
- * 生成非重复订单号,理论上限1毫秒1000个,可扩展
- * @param tname 测试用
- */
- public void makeOrderNum(String tname) {
- try {
- // 最终生成的订单号
- String finOrderNum = "";
- synchronized (lockObj) {
- // 取系统当前时间作为订单号变量前半部分,精确到毫秒
- long nowLong = Long.parseLong(new SimpleDateFormat("yyyyMMddHHmmssSSS").format(new Date()));
- // 计数器到最大值归零,可扩展更大,目前1毫秒处理峰值1000个,1秒100万
- if (orderNumCount >= maxPerMSECSize) {
- orderNumCount = 0L;
- }
- //组装订单号
- String countStr=maxPerMSECSize +orderNumCount+"";
- finOrderNum=nowLong+countStr.substring(1);
- orderNumCount++;
- System.out.println(finOrderNum + "--" + Thread.currentThread().getName() + "::" + tname );
- // Thread.sleep(1000);
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
- }
-
- public static void main(String[] args) {
- // 测试多线程调用订单号生成工具
- try {
- for (int i = 0; i < 200; i++) {
- Thread t1 = new Thread(new Runnable() {
- public void run() {
- MakeOrderNum makeOrder = new MakeOrderNum();
- makeOrder.makeOrderNum("a");
- }
- }, "at" + i);
- t1.start();
-
- Thread t2 = new Thread(new Runnable() {
- public void run() {
- MakeOrderNum makeOrder = new MakeOrderNum();
- makeOrder.makeOrderNum("b");
- }
- }, "bt" + i);
- t2.start();
- }
- } catch (Exception e) {
- e.printStackTrace();
- }
-
- }
-
- }
|