uuid.hex主键生成器

来源:互联网 发布:淘宝网店店铺介绍范文 编辑:程序博客网 时间:2024/05/22 12:22


对于数据库主键生成策略,大家都了解一些,尤其是关于hibernate的主键生成更是方便很多。

而大多数人只知道使用,hibernate给定的生成策略,我今天想说的是主键生成器,就是自己写一个类来生成主键。

在开始之前,先对hibernate给定的生成策略做个了解。



**************************************************************************************************

1  assigned:主键由外部程序负责生成,无需hibernate参与。特点是:主键的生成完全由用户决定,与底层数据库无关,用户需要维护主键值,并且要在session.save()之前指定主键值,否则会抛出异常。

2  hilo:通过hilo算法实现的主键生成机制,需要额外的数据库表保存主键生成历史状态。使用高低位算法生成主键,高低位算法使用一个高位值和一个低位值,然后把算法得到的两个值拼接起来作为数据库中的唯一主键。默认情况下采用的表是hibernate_unique_key。

3  seqhilo:与hilo类似,通过hilo算法实现的主键生成机制,只是主键历史状态保存在Sequence中,适用于支持Sequence的数据库

4  increment:主键按数值顺序递增。当HIbernate准备在数据库中插入一条记录时,首先从数据库表中取出当前主键字段的最大值,然后在最大值的基础上加1,作为当前持久化对象的标识符属性值。这种方式可能产生的问题是:如果当前有多个实例访问数据库,那么由于各个实例各自维持着主键状态,不同实例可能生成相同的主键,从而造成主键重复异常。

5  identity:采用数据库提供的主键生成机制

6  sequence:采用数据库提供的sequence生成主键,要设定序列名  <param name="sequence">name_seq</param>。如果未指定序列名,则hibernate默认使用名为“hibernate_sequence"的序列

7  native:有Hibernate根据地层数据库自行判断采用identity,hilo,sequence中一种作为主键生成机制

8  uuid.hex:由hibernate基于128位唯一值产生算法生成十六进制数值作为主键

9  uuid.string:与uuid.hex相似,只是生成的主键没有进行编码(16位),在某些数据库中可能出现问题

 10  foreign:使用外部表的字段作为主键

11  select:使用触发器生成主键

**************************************************************************************************

现在我想说的是UUID主键生成。

在hibernate的映射文件中,配置成<generator class="uuid"/>就可以直接使用了。

关于它,要知道几点知识:


一、用一个128-bit的UUID算法生成字符串类型的标识符。

二、在一个网络中唯一(生成算法使用了IP地址)。

三、UUID被编码为一个32位16进制数字的字符串。


好了,现在我们想自己写一个UUID主键生成器该怎么办呢?

其实很简单,既然Hibernate已经帮我们写好了,我们就可以看看它的源码怎么写的,单独把相关代码摘出来就可以了。


在hibernate3.jar中,UUIDHexGenerator

/*jadclipse*/// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.package org.hibernate.id;import java.io.PrintStream;import java.io.Serializable;import java.util.Properties;import org.hibernate.Hibernate;import org.hibernate.dialect.Dialect;import org.hibernate.engine.SessionImplementor;import org.hibernate.type.Type;import org.hibernate.util.PropertiesHelper;// Referenced classes of package org.hibernate.id://            AbstractUUIDGenerator, Configurable, IdentifierGeneratorpublic class UUIDHexGenerator extends AbstractUUIDGenerator    implements Configurable{    public UUIDHexGenerator()    {        sep = "";    }    protected String format(int intval)    {        String formatted = Integer.toHexString(intval);        StringBuffer buf = new StringBuffer("00000000");        buf.replace(8 - formatted.length(), 8, formatted);        return buf.toString();    }    protected 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 Serializable generate(SessionImplementor session, Object obj)    {        return (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 void configure(Type type, Properties params, Dialect d)    {        sep = PropertiesHelper.getString("separator", params, "");    }    public static void main(String args[])        throws Exception    {        Properties props = new Properties();        props.setProperty("separator", "/");        IdentifierGenerator gen = new UUIDHexGenerator();        ((Configurable)gen).configure(Hibernate.STRING, props, null);        IdentifierGenerator gen2 = new UUIDHexGenerator();        ((Configurable)gen2).configure(Hibernate.STRING, props, null);        for(int i = 0; i < 10; i++)        {            String id = (String)gen.generate(null, null);            System.out.println(id);            String id2 = (String)gen2.generate(null, null);            System.out.println(id2);        }    }    private String sep;}/*DECOMPILATION REPORTDecompiled from: D:\ChinaDevelopmentBankJBPM\workSpace\frame\webapp\WEB-INF\lib\hibernate3.jarTotal time: 195 msJad reported messages/errors:Exit status: 0Caught exceptions:*/

这个类继承了AbstractUUIDGenerator

/*jadclipse*/// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.package org.hibernate.id;import java.net.InetAddress;import org.hibernate.util.BytesHelper;// Referenced classes of package org.hibernate.id://            IdentifierGeneratorpublic abstract class AbstractUUIDGenerator    implements IdentifierGenerator{    public AbstractUUIDGenerator()    {    }    protected int getJVM()    {        return JVM;    }    protected short getCount()    {        Class class1 = org.hibernate.id.AbstractUUIDGenerator.class;        JVM INSTR monitorenter ;        if(counter < 0)            counter = 0;        return counter++;        Exception exception;        exception;        throw exception;    }    protected int getIP()    {        return IP;    }    protected short getHiTime()    {        return (short)(int)(System.currentTimeMillis() >>> 32);    }    protected int getLoTime()    {        return (int)System.currentTimeMillis();    }    private static final int IP;    private static short counter = 0;    private static final int JVM = (int)(System.currentTimeMillis() >>> 8);    static     {        int ipadd;        try        {            ipadd = BytesHelper.toInt(InetAddress.getLocalHost().getAddress());        }        catch(Exception e)        {            ipadd = 0;        }        IP = ipadd;    }}/*DECOMPILATION REPORTDecompiled from: D:\ChinaDevelopmentBankJBPM\workSpace\frame\webapp\WEB-INF\lib\hibernate3.jarTotal time: 156 msJad reported messages/errors:Overlapped try statements detected. Not all exception handlers will be resolved in the method getCountCouldn't fully decompile method getCountCouldn't resolve all exception handlers in method getCountExit status: 0Caught exceptions:*/

要想摘出来代码,要明白:继承的父类是必须要摘的,而接口我们完全可以忽略。

再把父类中的方法摘出放到子类中去,这样这个父类也没有利用价值了。

到此基本上快完成了,对于几个出错的方法,我们尽可能注释掉,最后不出错,写个main方法测试一下。



这是最后摘出来整合好的UUIDHexGenerator

package org.base.pk;import java.io.Serializable;import java.net.InetAddress;public class UUIDHexGenerator {public UUIDHexGenerator() {sep = "";}protected String format(int intval) {String formatted = Integer.toHexString(intval);StringBuffer buf = new StringBuffer("00000000");buf.replace(8 - formatted.length(), 8, formatted);return buf.toString();}protected 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 Serializable generate() {return (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();}private String sep;// /////////////////protected int getJVM() {return JVM;}protected short getCount() {Class class1 = org.base.pk.UUIDHexGenerator.class;if (counter < 0)counter = 0;return counter++;}protected int getIP() {return IP;}protected short getHiTime() {return (short) (int) (System.currentTimeMillis() >>> 32);}protected int getLoTime() {return (int) System.currentTimeMillis();}private static final int IP;private static short counter = 0;private static final int JVM = (int) (System.currentTimeMillis() >>> 8);static {int ipadd;try {ipadd = toInt(InetAddress.getLocalHost().getAddress());} catch (Exception e) {ipadd = 0;}IP = ipadd;}public static int toInt(byte bytes[])    {        int result = 0;        for(int i = 0; i < 4; i++)            result = ((result << 8) - -128) + bytes[i];        return result;    }public static void main(String args[]) throws Exception {UUIDHexGenerator gen = new UUIDHexGenerator();for (int i = 0; i < 10; i++) {String id = (String) gen.generate();System.out.println(id);}}}

运行一下,打印:

8ab78fab3cf123ac013cf123acb00000
8ab78fab3cf123ac013cf123acb10001
8ab78fab3cf123ac013cf123acb10002
8ab78fab3cf123ac013cf123acb10003
8ab78fab3cf123ac013cf123acb10004
8ab78fab3cf123ac013cf123acb10005
8ab78fab3cf123ac013cf123acb10006
8ab78fab3cf123ac013cf123acb10007
8ab78fab3cf123ac013cf123acb10008
8ab78fab3cf123ac013cf123acb10009


到此为止,还不算大功告成,毕竟这是一个主键生成器,算是工具类了。

我们要把这个类改成静态的,使用起来才算方便。

改变时注意变量sep要赋值空字符串,否则这个变量始终是null。

这是最终版的UUID主键生成器

package org.base.pk;import java.io.Serializable;import java.net.InetAddress;public class UUIDHexGenerator {public UUIDHexGenerator() {}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 Serializable generate() {return (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 String sep="";// /////////////////public static int getJVM() {return JVM;}public static short getCount() {Class class1 = org.base.pk.UUIDHexGenerator.class;if (counter < 0)counter = 0;return counter++;}public static int getIP() {return IP;}public static short getHiTime() {return (short) (int) (System.currentTimeMillis() >>> 32);}public static int getLoTime() {return (int) System.currentTimeMillis();}private static final int IP;private static short counter = 0;private static final int JVM = (int) (System.currentTimeMillis() >>> 8);static {int ipadd;try {ipadd = toInt(InetAddress.getLocalHost().getAddress());} catch (Exception e) {ipadd = 0;}IP = ipadd;}public static int toInt(byte bytes[])    {        int result = 0;        for(int i = 0; i < 4; i++)            result = ((result << 8) - -128) + bytes[i];        return result;    }public static void main(String args[]) throws Exception {for (int i = 0; i < 10; i++) {String id = (String) UUIDHexGenerator.generate();System.out.println(id);}}}

运行后打印:

8ab78fab3cf12cc3013cf12cc4020000
8ab78fab3cf12cc3013cf12cc4020001
8ab78fab3cf12cc3013cf12cc4020002
8ab78fab3cf12cc3013cf12cc4020003
8ab78fab3cf12cc3013cf12cc4020004
8ab78fab3cf12cc3013cf12cc4020005
8ab78fab3cf12cc3013cf12cc4020006
8ab78fab3cf12cc3013cf12cc4020007
8ab78fab3cf12cc3013cf12cc4020008
8ab78fab3cf12cc3013cf12cc4030009

大功告成。