唯一序列号生成,自测支持高并发,不支持集群

序列号组成:前缀 + 日期 + 尾数 

例如:ZC20140806000001

总共两个组成:一个枚举类,一个静态生成器,另外需要设计自动任务每日凌晨或其他时间重置一次尾数。


先上枚举类:

package com.boxmeal.base.constant.common;

import java.util.concurrent.atomic.AtomicInteger;
/**
 * 序列号生成枚举
 * @author bling
 *
 */
public enum SNEnum {

	// 订单编号
	SUIT("tcd",new AtomicInteger(0));
	
	private String prefix;
	
	private AtomicInteger count;
	
	private SNEnum(String prefix,AtomicInteger count) {
		this.prefix = prefix;
		this.count = count;
	}

	public void setPrefix(String prefix) {
		this.prefix = prefix;
	}

	public void setCount(AtomicInteger count) {
		this.count = count;
	}

	public String getPrefix() {
		return prefix;
	}

	public AtomicInteger getCount() {
		return count;
	}
}
枚举类是为了控制参数保证统一性。


再上生成器类:

package com.boxmeal.base.util;

import java.text.DecimalFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.concurrent.atomic.AtomicInteger;

import com.boxmeal.base.constant.common.SNEnum;

/**
 * 序列号生成器,每日自动从1开始增长
 * @author bling
 *
 */
public class SNGenerater {

	/**
	 * 日期格式化
	 */
	private static final SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMdd");
	
	/**
	 * 数字格式化
	 */
	private static final DecimalFormat decimalFormat = new DecimalFormat("000000");
	
	/**
	 * 序列号产生器
	 * @param sn
	 * @return
	 */
	public final static String genSuitSN(SNEnum sn){
		String prefix = sn.getPrefix();
		String dateStr = SNGenerater.dateFormat.format(new Date());
		String suffix = SNGenerater.decimalFormat.format(sn.getCount().incrementAndGet());
		return prefix + dateStr + suffix;
	}
	
	/**
	 * 重置序列号生成器:自动任务每日夜间00:00调用一次
	 */
	public final static void reset(){
		SNEnum[] snEnumAry = SNEnum.values();
		for(int i=0;i<snEnumAry.length;i++){
			snEnumAry[i].setCount(new AtomicInteger(0));
		}
	}
	
	public static void main(String[] args) {
		for(int i=0;i<100;i++){
			System.out.println(SNGenerater.genSuitSN(SNEnum.SUIT));
		}
		SNGenerater.reset();
		for(int i=0;i<100;i++){
			System.out.println(SNGenerater.genSuitSN(SNEnum.SUIT));
		}
	}
	
}

虽然在生成器方法没有加同步关键词,是因为尾数自增是原子操作。


测试类:

package com.boxmeal.test;

import com.boxmeal.base.constant.common.SNEnum;
import com.boxmeal.base.util.SNGenerater;

public class SNTest {

	class VisitThread implements Runnable{
		
		@Override
		public void run() {
			for(int i=0;i<100;i++){
				System.out.println(SNGenerater.genSuitSN(SNEnum.SUIT));
				try {
					Thread.sleep(20);
				} catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
		}
	}
	
	public void test(){
		for(int i=0;i<20;i++){
			new Thread(new VisitThread()).start();
		}
	}
	
	public static void main(String[] args) {
		new SNTest().test();
	}
	
}

开了20个线程并发2000次访问,测试结果没有发生重复。


  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
public synchronized String nextId() { long timestamp = timeGen(); //获取当前毫秒数 //如果服务器时间有问题(时钟后退) 报错。 if (timestamp < lastTimestamp) { throw new RuntimeException(String.format( "Clock moved backwards. Refusing to generate id for %d milliseconds", lastTimestamp - timestamp)); } //如果上次生成时间和当前时间相同,在同一毫秒内 if (lastTimestamp == timestamp) { //sequence自增,因为sequence只有12bit,所以和sequenceMask相与一下,去掉高位 sequence = (sequence + 1) & sequenceMask; //判断是否溢出,也就是每毫秒内超过4095,当为4096时,与sequenceMask相与,sequence就等于0 if (sequence == 0) { timestamp = tilNextMillis(lastTimestamp); //自旋等待到下一毫秒 } } else { sequence = 0L; //如果和上次生成时间不同,重置sequence,就是下一毫秒开始,sequence计数重新从0开始累加 } lastTimestamp = timestamp; long suffix = (datacenterId << datacenterIdShift) | (workerId << workerIdShift) | sequence; String datePrefix = DateFormatUtils.format(timestamp, "yyyyMMddHHMMssSSS"); return datePrefix + suffix; } protected long tilNextMillis(long lastTimestamp) { long timestamp = timeGen(); while (timestamp <= lastTimestamp) { timestamp = timeGen(); } return timestamp; } protected long timeGen() { return System.currentTimeMillis(); } private byte getLastIP(){ byte lastip = 0; try{ InetAddress ip = InetAddress.getLocalHost(); byte[] ipByte = ip.getAddress(); lastip = ipByte[ipByte.length - 1]; } catch (UnknownHostException e) { e.printStackTrace(); } return lastip; }
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值