什麼是uuid以及uuid在java中的使用

什麼是UUID?

UUID是Universally Unique Identifier的縮寫,它是在必定的範圍內(從特定的名字空間到全球)惟一的機器生成的標識符。UUID具備如下涵義:

html

  • 經由必定的算法機器生成

爲了保證UUID的惟一性,規範定義了包括網卡MAC地址、時間戳、名字空間(Namespace)、隨機或僞隨機數、時序等元素,以及從這些元素生成UUID的算法。UUID的複雜特性在保證了其惟一性的同時,意味着只能由計算機生成。java

  • 非人工指定,非人工識別

UUID是不能人工指定的,除非你冒着UUID重複的風險。UUID的複雜性決定了「通常人「不能直接從一個UUID知道哪一個對象和它關聯。算法

  • 在特定的範圍內重複的可能性極小

UUID的生成規範定義的算法主要目的就是要保證其惟一性。但這個惟一性是有限的,只在特定的範圍內才能獲得保證,這和UUID的類型有關(參見UUID的版本)。

UUID是16字節128位長的數字,一般以36字節的字符串表示,示例以下:

3F2504E0-4F89-11D3-9A0C-0305E82C3301

其中的字母是16進製表示,大小寫無關。

GUID(Globally Unique Identifier)是UUID的別名;但在實際應用中,GUID一般是指微軟實現的UUID。

UUID的版本

UUID具備多個版本,每一個版本的算法不一樣,應用範圍也不一樣。

首先是一個特例--Nil UUID--一般咱們不會用到它,它是由全爲0的數字組成,以下:

00000000-0000-0000-0000-000000000000

UUID Version 1:基於時間的UUID

基於時間的UUID經過計算當前時間戳、隨機數和機器MAC地址獲得。因爲在算法中使用了MAC地址,這個版本的UUID能夠保證在全球範圍的惟一性。但與此同時,使用MAC地址會帶來安全性問題,這就是這個版本UUID受到批評的地方。若是應用只是在局域網中使用,也可使用退化的算法,以IP地址來代替MAC地址--Java的UUID每每是這樣實現的(固然也考慮了獲取MAC的難度)。

UUID Version 2:DCE安全的UUID

DCE(Distributed Computing Environment)安全的UUID和基於時間的UUID算法相同,但會把時間戳的前4位置換爲POSIX的UID或GID。這個版本的UUID在實際中較少用到。

UUID Version 3:基於名字的UUID(MD5)

基於名字的UUID經過計算名字和名字空間的MD5散列值獲得。這個版本的UUID保證了:相同名字空間中不一樣名字生成的UUID的惟一性;不一樣名字空間中的UUID的惟一性;相同名字空間中相同名字的UUID重複生成是相同的。

UUID Version 4:隨機UUID

根據隨機數,或者僞隨機數生成UUID。這種UUID產生重複的機率是能夠計算出來的,但隨機的東西就像是買彩票:你期望它發財是不可能的,但狗屎運一般會在不經意中到來。

UUID Version 5:基於名字的UUID(SHA1)

和版本3的UUID算法相似,只是散列值計算使用SHA1(Secure Hash Algorithm 1)算法。

UUID的應用

從UUID的不一樣版本能夠看出,Version 1/2適合應用於分佈式計算環境下,具備高度的惟一性;Version 3/5適合於必定範圍內名字惟一,且須要或可能會重複生成UUID的環境下;至於Version 4,我我的的建議是最好不用(雖然它是最簡單最方便的)。

一般咱們建議使用UUID來標識對象或持久化數據,但如下狀況最好不使用UUID:

數據庫

  • 映射類型的對象。好比只有代碼及名稱的代碼表。
  • 人工維護的非系統生成對象。好比系統中的部分基礎數據。

對於具備名稱不可重複的天然特性的對象,最好使用Version 3/5的UUID。好比系統中的用戶。若是用戶的UUID是Version 1的,若是你不當心刪除了再重建用戶,你會發現人仍是那我的,用戶已經不是那個用戶了。(雖然標記爲刪除狀態也是一種解決方案,但會帶來實現上的複雜性。)

UUID生成器

我沒想着有人看完了這篇文章就去本身實現一個UUID生成器,因此前面的內容並不涉及算法的細節。下面是一些可用的Java UUID生成器:

apache

  • Java UUID Generator (JUG):開源UUID生成器,LGPL協議,支持MAC地址。
  • UUID:特殊的License,有源碼。
  • Java 5以上版本中自帶的UUID生成器:好像只能生成Version 3/4的UUID。

此外,Hibernate中也有一個UUID生成器,可是,生成的不是任何一個(規範)版本的UUID,強烈不建議使用。數組

——————————————————————安全

 

JAVA UUID 生成
 
GUID是一個128位長的數字,通常用16進製表示。算法的核心思想是結合機器的網卡、當地時間、一個隨即數來生成GUID。從理論上講,若是一臺機器每秒產生10000000個GUID,則能夠保證(機率意義上)3240年不重複。
 
UUID是1.5中新增的一個類,在java.util下,用它能夠產生一個號稱全球惟一的ID
 
package com.mytest;
 
import java.util.UUID;
 
public class UTest {
    public static void main(String[] args) {
        UUID uuid = UUID.randomUUID();
        System.out.println(uuid);
    }
}
 
 
UUID(Universally Unique Identifier)全局惟一標識符,是指在一臺機器上生成的數字,它保證對在同一時空中的全部機器都是惟一的。按照開放軟件基金會(OSF)制定的標準計算,用到了以太網卡地址、納秒級時間、芯片ID碼和許多可能的數字。由如下幾部分的組合:當前日期和時間(UUID的第一個部分與時間有關,若是你在生成一個UUID以後,過幾秒又生成一個UUID,則第一個部分不一樣,其他相同),時鐘序列,全局惟一的IEEE機器識別號(若是有網卡,從網卡得到,沒有網卡以其餘方式得到),UUID的惟一缺陷在於生成的結果串會比較長。 
在Java中生成UUID主要有如下幾種方式: 
 
JDK1.5 
若是使用的JDK1.5的話,那麼生成UUID變成了一件簡單的事,覺得JDK實現了UUID: 
java.util.UUID,直接調用便可. 
UUID uuid  =  UUID.randomUUID(); 
String s = UUID.randomUUID().toString();//用來生成數據庫的主鍵id很是不錯。。   
 
UUID是由一個十六位的數字組成,表現出來的形式例如 
550E8400-E29B-11D4-A716-446655440000   
 
//下面就是實現爲數據庫獲取一個惟一的主鍵id的代碼 
public class UUIDGenerator { 
    public UUIDGenerator() { 
    } 
    /** 
     * 得到一個UUID 
     * @return String UUID 
     */ 
    public static String getUUID(){ 
        String s = UUID.randomUUID().toString(); 
        //去掉「-」符號 
        return s.substring(0,8)+s.substring(9,13)+s.substring(14,18)+s.substring(19,23)+s.substring(24); 
    } 
    /** 
     * 得到指定數目的UUID 
     * @param number int 須要得到的UUID數量 
     * @return String[] UUID數組 
     */ 
    public static String[] getUUID(int number){ 
        if(number < 1){ 
            return null; 
        } 
        String[] ss = new String[number]; 
        for(int i=0;i<number;i++){ 
            ss[i] = getUUID(); 
        } 
        return ss; 
    } 
    public static void main(String[] args){ 
        String[] ss = getUUID(10); 
        for(int i=0;i<ss.length;i++){ 
            System.out.println(ss[i]); 
        } 
    } 
}   
 
 
兩種方式生成guid 與uuid
 
須要comm log 庫
 
/**
 * @author Administrator
 *
 * TODO To change the template for this generated type comment go to
 * Window - Preferences - Java - Code Style - Code Templates
 */
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.security.SecureRandom;
import java.util.Random;
 
public class RandomGUID extends Object {
   protected final org.apache.commons.logging.Log logger = org.apache.commons.logging.LogFactory
      .getLog(getClass());
 
   public String valueBeforeMD5 = "";
   public String valueAfterMD5 = "";
   private static Random myRand;
   private static SecureRandom mySecureRand;
 
   private static String s_id;
   private static final int PAD_BELOW = 0x10;
   private static final int TWO_BYTES = 0xFF;
 
   /*
    * Static block to take care of one time secureRandom seed.
    * It takes a few seconds to initialize SecureRandom.  You might
    * want to consider removing this static block or replacing
    * it with a "time since first loaded" seed to reduce this time.
    * This block will run only once per JVM instance.
      */
 
   static {
      mySecureRand = new SecureRandom();
      long secureInitializer = mySecureRand.nextLong();
      myRand = new Random(secureInitializer);
      try {
         s_id = InetAddress.getLocalHost().toString();
      } catch (UnknownHostException e) {
         e.printStackTrace();
      }
 
   }
 
 
   /*
    * Default constructor.  With no specification of security option,
    * this constructor defaults to lower security, high performance.
    */
   public RandomGUID() {
      getRandomGUID(false);
   }
 
   /*
    * Constructor with security option.  Setting secure true
    * enables each random number generated to be cryptographically
    * strong.  Secure false defaults to the standard Random function seeded
    * with a single cryptographically strong random number.
    */
   public RandomGUID(boolean secure) {
      getRandomGUID(secure);
   }
 
   /*
    * Method to generate the random GUID
    */
   private void getRandomGUID(boolean secure) {
      MessageDigest md5 = null;
      StringBuffer sbValueBeforeMD5 = new StringBuffer(128);
 
      try {
         md5 = MessageDigest.getInstance("MD5");
      } catch (NoSuchAlgorithmException e) {
         logger.error("Error: " + e);
      }
 
      try {
         long time = System.currentTimeMillis();
         long rand = 0;
 
         if (secure) {
            rand = mySecureRand.nextLong();
         } else {
            rand = myRand.nextLong();
         }
         sbValueBeforeMD5.append(s_id);
         sbValueBeforeMD5.append(":");
         sbValueBeforeMD5.append(Long.toString(time));
         sbValueBeforeMD5.append(":");
         sbValueBeforeMD5.append(Long.toString(rand));
 
         valueBeforeMD5 = sbValueBeforeMD5.toString();
         md5.update(valueBeforeMD5.getBytes());
 
         byte[] array = md5.digest();
         StringBuffer sb = new StringBuffer(32);
         for (int j = 0; j < array.length; ++j) {
            int b = array[j] & TWO_BYTES;
            if (b < PAD_BELOW)
               sb.append('0');
            sb.append(Integer.toHexString(b));
         }
 
         valueAfterMD5 = sb.toString();
 
      } catch (Exception e) {
         logger.error("Error:" + e);
      }
   }
 
   /*
    * Convert to the standard format for GUID
    * (Useful for SQL Server UniqueIdentifiers, etc.)
    * Example: C2FEEEAC-CFCD-11D1-8B05-00600806D9B6
    */
   public String toString() {
      String raw = valueAfterMD5.toUpperCase();
      StringBuffer sb = new StringBuffer(64);
      sb.append(raw.substring(0, 8));
      sb.append("-");
      sb.append(raw.substring(8, 12));
      sb.append("-");
      sb.append(raw.substring(12, 16));
      sb.append("-");
      sb.append(raw.substring(16, 20));
      sb.append("-");
      sb.append(raw.substring(20));
 
      return sb.toString();
   }
 
 
     // Demonstraton and self test of class
     public static void main(String args[]) {
       for (int i=0; i< 100; i++) {
         RandomGUID myGUID = new RandomGUID();
         System.out.println("Seeding String=" + myGUID.valueBeforeMD5);
         System.out.println("rawGUID=" + myGUID.valueAfterMD5);
         System.out.println("RandomGUID=" + myGUID.toString());
       }
     }
 
 
}
 
一樣
 
UUID uuid = UUID.randomUUID();
System.out.println("{"+uuid.toString()+"}");
 
UUID是指在一臺機器上生成的數字,它保證對在同一時空中的全部機器都是惟一的。一般平臺會提供生成UUID的API。UUID按照開放軟件基金會(OSF)制定的標準計算,用到了以太網卡地址、納秒級時間、芯片ID碼和許多可能的數字。由如下幾部分的組合:當前日期和時間(UUID的第一個部分與時間有關,若是你在生成一個UUID以後,過幾秒又生成一個UUID,則第一個部分不一樣,其他相同),時鐘序列,全局惟一的IEEE機器識別號(若是有網卡,從網卡得到,沒有網卡以其餘方式得到),UUID的惟一缺陷在於生成的結果串會比較長。關於UUID這個標準使用最廣泛的是微軟的GUID(Globals Unique Identifiers)。
 
UUID含義是通用惟一識別碼 (Universally Unique Identifier),這 是一個軟件建構的標準,也是被開源軟件基金會 (Open Software Foundation, OSF) 的組織在分佈式計算環境 (Distributed Computing Environment, DCE) 領域的一部份。UUID 的目的,是讓分佈式系統中的全部元素,都能有惟一的辨識資訊,而不須要透過中央控制端來作辨識資訊的指定。如此一來,每一個人均可以創建不與其它人衝突的 UUID。在這樣的狀況下,就不需考慮數據庫創建時的名稱重複問題。目前最普遍應用的 UUID,便是微軟的 Microsoft's Globally Unique Identifiers (GUIDs),而其餘重要的應用,則有 Linux ext2/ext3 檔案系統、LUKS 加密分割區、GNOME、KDE、Mac OS X 等等。
 
如下是具體生成UUID的例子:
 
view plaincopy to clipboardprint?
package test;   
  
import java.util.UUID;   
  
public class UUIDGenerator {   
    public UUIDGenerator() {   
    }   
  
    public static String getUUID() {   
        UUID uuid = UUID.randomUUID();   
        String str = uuid.toString();   
        // 去掉"-"符號   
        String temp = str.substring(0, 8) + str.substring(9, 13) + str.substring(14, 18) + str.substring(19, 23) + str.substring(24);   
        return str+","+temp;   
    }   
    //得到指定數量的UUID   
    public static String[] getUUID(int number) {   
        if (number < 1) {   
            return null;   
        }   
        String[] ss = new String[number];   
        for (int i = 0; i < number; i++) {   
            ss[i] = getUUID();   
        }   
        return ss;   
    }   
  
    public static void main(String[] args) {   
        String[] ss = getUUID(10);   
        for (int i = 0; i < ss.length; i++) {   
            System.out.println("ss["+i+"]====="+ss[i]);   
        }   
    }   
}  
package test;
 
import java.util.UUID;
 
public class UUIDGenerator {
 public UUIDGenerator() {
 }
 
 public static String getUUID() {
  UUID uuid = UUID.randomUUID();
  String str = uuid.toString();
  // 去掉"-"符號
  String temp = str.substring(0, 8) + str.substring(9, 13) + str.substring(14, 18) + str.substring(19, 23) + str.substring(24);
  return str+","+temp;
 }
 //得到指定數量的UUID
 public static String[] getUUID(int number) {
  if (number < 1) {
   return null;
  }
  String[] ss = new String[number];
  for (int i = 0; i < number; i++) {
   ss[i] = getUUID();
  }
  return ss;
 }
 
 public static void main(String[] args) {
  String[] ss = getUUID(10);
  for (int i = 0; i < ss.length; i++) {
   System.out.println("ss["+i+"]====="+ss[i]);
  }
 }
}
 
結果:
 
view plaincopy to clipboardprint?
ss[0]=====4cdbc040-657a-4847-b266-7e31d9e2c3d9,4cdbc040657a4847b2667e31d9e2c3d9   
ss[1]=====72297c88-4260-4c05-9b05-d28bfb11d10b,72297c8842604c059b05d28bfb11d10b   
ss[2]=====6d513b6a-69bd-4f79-b94c-d65fc841ea95,6d513b6a69bd4f79b94cd65fc841ea95   
ss[3]=====d897a7d3-87a3-4e38-9e0b-71013a6dbe4c,d897a7d387a34e389e0b71013a6dbe4c   
ss[4]=====5709f0ba-31e3-42bd-a28d-03485b257c94,5709f0ba31e342bda28d03485b257c94   
ss[5]=====530fbb8c-eec9-48d1-ae1b-5f792daf09f3,530fbb8ceec948d1ae1b5f792daf09f3   
ss[6]=====4bf07297-65b2-45ca-b905-6fc6f2f39158,4bf0729765b245cab9056fc6f2f39158   
ss[7]=====6e5a0e85-b4a0-485f-be54-a758115317e1,6e5a0e85b4a0485fbe54a758115317e1   
ss[8]=====245accec-3c12-4642-967f-e476cef558c4,245accec3c124642967fe476cef558c4   
ss[9]=====ddd4b5a9-fecd-446c-bd78-63b70bb500a1,ddd4b5a9fecd446cbd7863b70bb500a1  
ss[0]=====4cdbc040-657a-4847-b266-7e31d9e2c3d9,4cdbc040657a4847b2667e31d9e2c3d9
ss[1]=====72297c88-4260-4c05-9b05-d28bfb11d10b,72297c8842604c059b05d28bfb11d10b
ss[2]=====6d513b6a-69bd-4f79-b94c-d65fc841ea95,6d513b6a69bd4f79b94cd65fc841ea95
ss[3]=====d897a7d3-87a3-4e38-9e0b-71013a6dbe4c,d897a7d387a34e389e0b71013a6dbe4c
ss[4]=====5709f0ba-31e3-42bd-a28d-03485b257c94,5709f0ba31e342bda28d03485b257c94
ss[5]=====530fbb8c-eec9-48d1-ae1b-5f792daf09f3,530fbb8ceec948d1ae1b5f792daf09f3
ss[6]=====4bf07297-65b2-45ca-b905-6fc6f2f39158,4bf0729765b245cab9056fc6f2f39158
ss[7]=====6e5a0e85-b4a0-485f-be54-a758115317e1,6e5a0e85b4a0485fbe54a758115317e1
ss[8]=====245accec-3c12-4642-967f-e476cef558c4,245accec3c124642967fe476cef558c4
ss[9]=====ddd4b5a9-fecd-446c-bd78-63b70bb500a1,ddd4b5a9fecd446cbd7863b70bb500a1
 
  能夠看出,UUID 是指在一臺機器上生成的數字,它保證對在同一時空中的全部機器都是惟一的。一般平臺會提供生成的API。按照開放軟件基金會(OSF)制定的標準計算,用到了以太網卡地址、納秒級時間、芯片ID碼和許多可能的數字
 
  UUID由如下幾部分的組合:
 
  (1)當前日期和時間,UUID的第一個部分與時間有關,若是你在生成一個UUID以後,過幾秒又生成一個UUID,則第一個部分不一樣,其他相同。
 
  (2)時鐘序列
 
  (3)全局惟一的IEEE機器識別號,若是有網卡,從網卡MAC地址得到,沒有網卡以其餘方式得到。
 
  UUID的惟一缺陷在於生成的結果串會比較長。關於UUID這個標準使用最廣泛的是微軟的GUID(Globals Unique Identifiers)。在ColdFusion中能夠用CreateUUID()函數很簡單的生成UUID,其格式爲:xxxxxxxx-xxxx- xxxx-xxxxxxxxxxxxxxxx(8-4-4-16),其中每一個 x 是 0-9 或 a-f 範圍內的一個十六進制的數字。而標準的UUID格式爲:xxxxxxxx-xxxx-xxxx-xxxxxx-xxxxxxxxxx (8-4-4-4-12),能夠從cflib 下載CreateGUID() UDF進行轉換。
 
  使用UUID的好處在分佈式的軟件系統中(好比:DCE/RPC, COM+,CORBA)就能體現出來,它能保證每一個節點所生成的標識都不會重複,而且隨着WEB服務等整合技術的發展,UUID的優點將更加明顯。根據使用的特定機制,UUID不只須要保證是彼此不相同的,或者最少也是與公元3400年以前其餘任何生成的通用唯一標識符有很是大的區別。
 
  通用唯一標識符還能夠用來指向大多數的可能的物體。微軟和其餘一些軟件公司都傾向使用全球唯一標識符(GUID),這也是通用唯一標識符的一種類型,可用來指向組建對象模塊對象和其餘的軟件組件。第一個通用唯一標識符是在網羅計算機系統(NCS)中建立,而且隨後成爲開放軟件基金會(OSF)的分佈式計算環境(DCE)的組件。
 
 
——————————————————
轉載自:http://blog.csdn.net/lidew521/article/details/8522696
http://blog.csdn.net/yipiankongbai/article/details/25243531   
http://www.blogjava .NET/feelyou/archive/2008/10/14/234320.html
相關文章
相關標籤/搜索