2016-10-25 2 views
4

В питона, чтобы сделать тип 5 UUID можно просто сделать:Как сделать UUID типа 5 в Java?

import uuid 
print uuid.uuid5(uuid.NAMESPACE_URL, 'my string') 

Просматривая java documentation for java.util.UUID, я не вижу, как это сделать. Во-первых, тип 5 не упоминается. У них есть тип 3, но подпись:

nameUUIDFromBytes(byte[] name) 
Static factory to retrieve a type 3 (name based) UUID based on the specified byte array. 

Как мы можем сделать тип 5 UUID в Java?

+0

Может быть, это помогает https://modules.ceylon-lang.org/repo/1/herd/uuid/0.0.10/module-doc/api/index.html –

+0

Возможные дубликата [ эта реализация Java UUID5 не проходит единичный тест] (h ttp: //stackoverflow.com/questions/16501357/this-java-uuid5-implementation-not-passing-unit-test) – MordechayS

ответ

6

Вы можете реализовать его самостоятельно, выполнив код, предложенный в https://stackoverflow.com/a/28776880/1452094. Однако это требует некоторого ворчания, поскольку конструктор j.u.UUID занимает много времени.

С Java 8 стандартная библиотека, похоже, не поддерживает тип 5. Но сторонние библиотеки, такие как «Apache Commons Id», имеют реализации UUID, которые его поддерживают.

EDIT: Вот полностью функциональная реализация:

import java.nio.charset.Charset; 
import java.security.MessageDigest; 
import java.security.NoSuchAlgorithmException; 
import java.util.Objects; 
import java.util.UUID; 

public class UUIDType5 { 
    private static final Charset UTF8 = Charset.forName("UTF-8"); 
    public static final UUID NAMESPACE_DNS = UUID.fromString("6ba7b810-9dad-11d1-80b4-00c04fd430c8"); 
    public static final UUID NAMESPACE_URL = UUID.fromString("6ba7b811-9dad-11d1-80b4-00c04fd430c8"); 
    public static final UUID NAMESPACE_OID = UUID.fromString("6ba7b812-9dad-11d1-80b4-00c04fd430c8"); 
    public static final UUID NAMESPACE_X500 = UUID.fromString("6ba7b814-9dad-11d1-80b4-00c04fd430c8"); 

    public static UUID nameUUIDFromNamespaceAndString(UUID namespace, String name) { 
     return nameUUIDFromNamespaceAndBytes(namespace, Objects.requireNonNull(name, "name == null").getBytes(UTF8)); 
    } 

    public static UUID nameUUIDFromNamespaceAndBytes(UUID namespace, byte[] name) { 
     MessageDigest md; 
     try { 
      md = MessageDigest.getInstance("SHA-1"); 
     } catch (NoSuchAlgorithmException nsae) { 
      throw new InternalError("SHA-1 not supported"); 
     } 
     md.update(toBytes(Objects.requireNonNull(namespace, "namespace is null"))); 
     md.update(Objects.requireNonNull(name, "name is null")); 
     byte[] sha1Bytes = md.digest(); 
     sha1Bytes[6] &= 0x0f; /* clear version  */ 
     sha1Bytes[6] |= 0x50; /* set to version 5  */ 
     sha1Bytes[8] &= 0x3f; /* clear variant  */ 
     sha1Bytes[8] |= 0x80; /* set to IETF variant */ 
     return fromBytes(sha1Bytes); 
    } 

    private static UUID fromBytes(byte[] data) { 
     // Based on the private UUID(bytes[]) constructor 
     long msb = 0; 
     long lsb = 0; 
     assert data.length >= 16; 
     for (int i = 0; i < 8; i++) 
      msb = (msb << 8) | (data[i] & 0xff); 
     for (int i = 8; i < 16; i++) 
      lsb = (lsb << 8) | (data[i] & 0xff); 
     return new UUID(msb, lsb); 
    } 

    private static byte[] toBytes(UUID uuid) { 
     // inverted logic of fromBytes() 
     byte[] out = new byte[16]; 
     long msb = uuid.getMostSignificantBits(); 
     long lsb = uuid.getLeastSignificantBits(); 
     for (int i = 0; i < 8; i++) 
      out[i] = (byte) ((msb >> ((7 - i) * 8)) & 0xff); 
     for (int i = 8; i < 16; i++) 
      out[i] = (byte) ((lsb >> ((15 - i) * 8)) & 0xff); 
     return out; 
    } 
} 

Чтобы проверить это работает, я побежал следующий код:

public static void main(String[] args) { 
    UUID test = UUIDType5.nameUUIDFromNamespaceAndString(NAMESPACE_URL, "google.com"); 
    System.out.println(test); 
    System.out.println(test.version()); 
} 

Это создало вывод:

fedb2fa3- 8f5c-5189-80e6-f563dd1cb8f9

Проверенный против официальной реализации питона:

>>> печать (uuid.uuid5 (uuid.NAMESPACE_URL, 'google.com'))

fedb2fa3-8f5c-5189 -80e6-f563dd1cb8f9

+2

Спасибо, что посмотрю, катите меня. [У Apache Commons Id пока нет официальных релизов.] (Http://commons.apache.org/sandbox/commons-id/downloads.html) –

+0

То, что я получаю для быстрой проверки, если какой-либо из крупных поддержка поддерживаемых библиотек: P – Kiskae

+0

Вы можете обратиться к реализации UUID типа 3: http://grepcode.com/file/repository.grepcode.com/java/root/jdk/openjdk/6-b14/java/util /UUID.java#UUID.nameUUIDFromBytes%28byte%5B%5D%29 - Различия заключаются в том, что дайджест является «sha1», он сначала переваривает байты UUID пространства имен, а затем имя пространства имен и нажимает «md5Bytes [6] | = 0x30; 'должно быть' sha1Bytes [6] | = 0x50; ' – Kiskae

Смежные вопросы