首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > JAVA > J2EE开发 >

怎的生成唯一订单号

2013-09-09 
怎样生成唯一订单号比如格式为:yyyy-MM-dd + 00000001,yyyy-MM-dd + 00000002,yyyy-MM-dd + 00000003,...y

怎样生成唯一订单号
比如格式为:

yyyy-MM-dd + 00000001,
yyyy-MM-dd + 00000002,
yyyy-MM-dd + 00000003,
...

yyyy-MM-dd是订单产生日日期,后跟8位数字,每天都是从00000001开始。

大家都有什么好方法,能保证唯一性和速度。
[解决办法]
从年月日到时间 ,,取到毫秒
[解决办法]
上面说用时间来取唯一值不行吧,如果访问量很大,同一毫秒上来的订单那不是会重复了呢?

比较严谨的做法用一个自动增长的表来产生一个唯一值。

[解决办法]
办法有,有点小麻烦。 

首先定义一个工厂类,生成你的订单号序列。


package bean;

import java.text.DecimalFormat;
import java.text.SimpleDateFormat;
import java.util.Date;

import org.springframework.beans.factory.FactoryBean;


public class SequenceFactoryBean implements FactoryBean<String> {

private static long counter = 0;

public synchronized String getObject() throws Exception {
String date = new SimpleDateFormat("yyyy-MM-dd").format(new Date()) + " + ";
String sequ = new DecimalFormat("00000000").format(counter ++);
return date + sequ;
}

public Class<String> getObjectType() {
return String.class;
}

public boolean isSingleton() {
return false;
}

public static void reset() {
SequenceFactoryBean.counter = 0;
}
}


注意里面的静态方法 reset方法, 这个用来在每天00:00:00重置。
这个方法由任务调度框架Quartz调用。

首先定义一个job

package job;

import bean.SequenceFactoryBean;

public class SequenceResetJob {

public void execute() {
SequenceFactoryBean.reset();
}
}


配置Quartz,在每天零点触发触发器就可以了。 

Spring 的配置文件。 

<bean id="sequence" class="bean.SequenceFactoryBean" />

<bean id="job" class="job.SequenceResetJob" />

<bean id="jobDetail"
class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">


<property name="targetObject" ref="job" />
<property name="targetMethod" value="execute" />
</bean>

<bean id="cronTrigger" class="org.springframework.scheduling.quartz.CronTriggerBean">
<property name="jobDetail" ref="jobDetail" />
<property name="cronExpression" value="0 0 0 * * ?" />
</bean>

<bean name="quartzScheduler"
class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
<property name="triggers">
<list>
<ref bean="cronTrigger" />
</list>
</property>
</bean>


[解决办法]
你要用到序列号的话,注入一个就行了。 这个不是单例,每次都会生成一个新的订单号。

PS:此程序不适合服务器集群,如果是服务器集群的话,应该去查看数据库服务器的时间。而不是简单地 new Date();
[解决办法]
并发量大的话,简单的算法肯定会出现重复的订单号,俺收藏了一个类用于产生32位的绝对全球唯一的编号,类似于hibernate中uuid生成方式



package com.anxin.utils;

import java.io.Serializable;
import java.net.InetAddress;

/**
 * 生成类似hibernate中uuid 32位主键序列
 * 
 * @version: V1.0
 */
public class UUIDGenerator {

private static final int IP;

public static int IptoInt(byte[] bytes) {
int result = 0;
for (int i = 0; i < 4; i++) {
result = (result << 8) - Byte.MIN_VALUE + (int) bytes[i];
}
return result;
}

static {
int ipadd;
try {
ipadd = IptoInt(InetAddress.getLocalHost().getAddress());
} catch (Exception e) {
ipadd = 0;
}
IP = ipadd;
}
private static short counter = (short) 0;
private static final int JVM = (int) (System.currentTimeMillis() >>> 8);

public UUIDGenerator() {
}
public static int getJVM() {
return JVM;
}
public static short getCount() {
synchronized (UUIDGenerator.class) {
if (counter < 0)
counter = 0;
return counter++;
}
}
public static int getIP() {
return IP;
}
public static short getHiTime() {


return (short) (System.currentTimeMillis() >>> 32);
}

public static int getLoTime() {
return (int) System.currentTimeMillis();
}

private final static String sep = "";

public static String format(int intval) {
String formatted = Integer.toHexString(intval);
StringBuffer buf = new StringBuffer("00000000");
buf.replace(8 - formatted.length(), 8, formatted);
return buf.toString();
}

public  static String format(short shortval) {
String formatted = Integer.toHexString(shortval);
StringBuffer buf = new StringBuffer("0000");
buf.replace(4 - formatted.length(), 4, formatted);
return buf.toString();
}

public static String generate() {
return String.valueOf(new StringBuffer(36).append(format(getIP())).append(sep)
.append(format(getJVM())).append(sep)
.append(format(getHiTime())).append(sep)
.append(format(getLoTime())).append(sep)
.append(format(getCount())).toString());
}
public static void main(String args[]){
System.out.println(UUIDGenerator.generate());
}
}


[解决办法]
大约看了一下楼上的,挺精彩的。 

不过楼上的可知, JDK提供了UUID算法的。

UUID.randomUUID().toString().replaceAll("-", "");

[解决办法]
用户ID+时间戳。。何如?
[解决办法]
完全没有必要每天从 1 开始
[解决办法]
经验不足,关于集群无法给出跟详细的解答了。 等高手!
[解决办法]
前面取时间,最后加一个1000以内的随机数。
[解决办法]
你可以写一个方法,生成一定格式的号码,首先要读取数据库判断这个号码是否存在,不存在则写入,存在则加一写入,如果日期不一样从一开始
[解决办法]
由字母加数字随机生成个16位的或三十二位的更好
[解决办法]
只用时间不够,一毫秒可能生成若干个序列号。
前面加用户ID应该可以了。
一个操作ID在一个毫秒里肯定只有一个序列号。
如果一个操作要生成若干个序列号,那后面再加个自增变量就可以了。

userID+time+(i++)
[解决办法]
UUID是比较常用的,简单点的做法的话,就用数据库的序列来做流水号,前面再加上日期
------解决方案--------------------


引用:
比如格式为:

yyyy-MM-dd + 00000001,
yyyy-MM-dd + 00000002,
yyyy-MM-dd + 00000003,
...

yyyy-MM-dd是订单产生日日期,后跟8位数字,每天都是从00000001开始。

大家都有什么好方法,能保证唯一性和速度。


每次插入数据前 先max(id) 查询当前数据库数据最大的主键值
然后截取前面10位 判断是否是当天
如果是 则将后面的 00000003转化为数字 并+1 然后根据位数 补0
如果不是今天 那么 直接跟00000001

这样做最大的瓶颈是 假如你数据插入的比较频繁或者批量导入 这样就会造成重码
[解决办法]
时间戳 + 序列
时间戳可精确到毫秒,序列使用静态变量,写一个自加方法,一个重置方法,生成一个序列号是调用自加,达到最大重置。
[解决办法]
引用:
引用:
比如格式为:

yyyy-MM-dd + 00000001,
yyyy-MM-dd + 00000002,
yyyy-MM-dd + 00000003,
...

yyyy-MM-dd是订单产生日日期,后跟8位数字,每天都是从00000001开始。

大家都有什么好方法,能保证唯一性和速度。


每次插入数据前 先max(id) 查询……

重码问题,方面声明为 synchronized 的,并用一个全局变量 记录当前编号, 方法里面 ++ 可以解决

热点排行