Jak mogę wygenerować UUID w C #

137

Tworzę plik .idl programowo. Jak programowo utworzyć identyfikatory UUID dla interfejsów i metod.

Czy mogę programowo wygenerować UUID?

Uma Shankar Subramani
źródło
25
Masz na myśli Guid.NewGuid()?
Slaks

Odpowiedzi:

220

Prawdopodobnie szukasz System.Guid.NewGuid().

Tudor
źródło
33
Możesz także zrobić String UUID = Guid.NewGuid (). ToString ()
Justin
10
GUID i UUID są takie same?
Uma Shankar Subramani,
17
@Uma Shankar Subramani: GUID = globalnie unikalny identyfikator, UUID = uniwersalnie unikalny identyfikator. Różne słowa na tę samą koncepcję.
Tudor
1
Potrzebujesz sformatowania identyfikatora GUID jako ciągu innego niż domyślny, możesz użyć ToString(string format)przeciążenia, które akceptuje jeden z kilku specyfikatorów formatu.
Michiel van Oosterhout
7
Prawdopodobnie będziesz chciał to zrobić, System.Guid.NewGuid().ToString("B").ToUpper()jeśli chcesz być kompatybilny z niektórymi narzędziami MS Build, które nie rozumieją identyfikatorów UUID z małych liter. Na przykład vdprojprojekty instalacyjne mają identyfikatory UUID zapisane dużymi literami i zgłoszą wyjątek, jeśli nadasz mu małe litery.
Mark Lakata
43

Uważaj: podczas gdy reprezentacje ciągów dla .NET Guid i (RFC4122) UUID są identyczne, format przechowywania jest inny . .NET handluje bajtami little-endian dla pierwszych trzech Guidczęści.

Jeśli transmitujesz bajty (na przykład jako base64), nie możesz go po prostu użyć Guid.ToByteArray()i zakodować. Będziesz potrzebował Array.Reversepierwszych trzech części (Dane1-3).

Robię to w ten sposób:

var rfc4122bytes = Convert.FromBase64String("aguidthatIgotonthewire==");
Array.Reverse(rfc4122bytes,0,4);
Array.Reverse(rfc4122bytes,4,2);
Array.Reverse(rfc4122bytes,6,2);
var guid = new Guid(rfc4122bytes);

Zobacz tę odpowiedź, aby uzyskać szczegółowe informacje dotyczące implementacji platformy .NET.

Edycja : Podziękowania dla Jeffa Walkera, Code Ranger, za wskazanie, że elementy wewnętrzne nie są istotne dla formatu tablicy bajtów, która wchodzi i wychodzi z konstruktora tablicy bajtów i ToByteArray().

Ben Mosher
źródło
Uwaga: zdaję sobie sprawę, że OP prawdopodobnie oznaczał Guid(ponieważ jest przeznaczony dla .idl), ale właśnie to wpadłem. Więc proszę bardzo, Bingers i Googlers.
Ben Mosher,
1
Nie mogę tego przetestować, ale czy na pewno powinieneś sprawdzić BitConverter.IsLittleEndian zamiast po prostu zawsze cofać. Dokumentacja dla Guid.ToByteArray () wywołuje kolejność bajtów jako little endian i mówi, że konstruktor pasuje. Specyfikacja GUID to little endian. Myślę, że powinno to być niezależne od kolejności bajtów w komputerze.
Jeff Walker Code Ranger
@JeffWalkerCodeRanger: od Erica Lipperta, lata temu: blogs.msdn.com/b/ericlippert/archive/2004/05/25/141525.aspx
Ben Mosher
Nie rozumiem, jak link Eric Lippert odpowiada na pytanie. Patrząc na kod Mono pod adresem github.com/mono/mono/blob/master/mcs/class/corlib/System/ ... wydaje mi się, że zawsze zakładają, że bajty są w porządku little endian, niezależnie od natywnego endianness. To pasuje do mojego rozumienia, że ​​gdyby zależało od platformy, nie pasowałoby do semantyki interfejsu MS lub specyfikacji. Patrząc na zdemontowane mscorelib, wydaje się, że bajty w tablicy są również w porządku little endian.
Jeff Walker Code Ranger,
Wygląda na to, że masz rację. Chociaż format pamięci wewnętrznej jest wrażliwy na endianness, konstruktor i ToByteArraynie są; zawsze są lil'-endian. Edytuj przychodzące.
Ben Mosher
3

Oto rozwiązanie po stronie klienta „sekwencyjny identyfikator guid”.

http://www.pinvoke.net/default.aspx/rpcrt4.uuidcreate

using System;
using System.Runtime.InteropServices;


namespace MyCompany.MyTechnology.Framework.CrossDomain.GuidExtend
{
    public static class Guid
    {

        /*

        Original Reference for Code:
        http://www.pinvoke.net/default.aspx/rpcrt4/UuidCreateSequential.html

        */


        [DllImport("rpcrt4.dll", SetLastError = true)]
        static extern int UuidCreateSequential(out System.Guid guid);

        public static System.Guid NewGuid()
        {
            return CreateSequentialUuid();
        }


        public static System.Guid CreateSequentialUuid()
        {
            const int RPC_S_OK = 0;
            System.Guid g;
            int hr = UuidCreateSequential(out g);
            if (hr != RPC_S_OK)
                throw new ApplicationException("UuidCreateSequential failed: " + hr);
            return g;
        }


        /*

        Text From URL above:

        UuidCreateSequential (rpcrt4)

        Type a page name and press Enter. You'll jump to the page if it exists, or you can create it if it doesn't.
        To create a page in a module other than rpcrt4, prefix the name with the module name and a period.
        . Summary
        Creates a new UUID 
        C# Signature:
        [DllImport("rpcrt4.dll", SetLastError=true)]
        static extern int UuidCreateSequential(out Guid guid);


        VB Signature:
        Declare Function UuidCreateSequential Lib "rpcrt4.dll" (ByRef id As Guid) As Integer


        User-Defined Types:
        None.

        Notes:
        Microsoft changed the UuidCreate function so it no longer uses the machine's MAC address as part of the UUID. Since CoCreateGuid calls UuidCreate to get its GUID, its output also changed. If you still like the GUIDs to be generated in sequential order (helpful for keeping a related group of GUIDs together in the system registry), you can use the UuidCreateSequential function.

        CoCreateGuid generates random-looking GUIDs like these:

        92E60A8A-2A99-4F53-9A71-AC69BD7E4D75
        BB88FD63-DAC2-4B15-8ADF-1D502E64B92F
        28F8800C-C804-4F0F-B6F1-24BFC4D4EE80
        EBD133A6-6CF3-4ADA-B723-A8177B70D268
        B10A35C0-F012-4EC1-9D24-3CC91D2B7122



        UuidCreateSequential generates sequential GUIDs like these:

        19F287B4-8830-11D9-8BFC-000CF1ADC5B7
        19F287B5-8830-11D9-8BFC-000CF1ADC5B7
        19F287B6-8830-11D9-8BFC-000CF1ADC5B7
        19F287B7-8830-11D9-8BFC-000CF1ADC5B7
        19F287B8-8830-11D9-8BFC-000CF1ADC5B7



        Here is a summary of the differences in the output of UuidCreateSequential:

        The last six bytes reveal your MAC address 
        Several GUIDs generated in a row are sequential 
        Tips & Tricks:
        Please add some!

        Sample Code in C#:
        static Guid UuidCreateSequential()
        {
           const int RPC_S_OK = 0;
           Guid g;
           int hr = UuidCreateSequential(out g);
           if (hr != RPC_S_OK)
             throw new ApplicationException
               ("UuidCreateSequential failed: " + hr);
           return g;
        }



        Sample Code in VB:
        Sub Main()
           Dim myId As Guid
           Dim code As Integer
           code = UuidCreateSequential(myId)
           If code <> 0 Then
             Console.WriteLine("UuidCreateSequential failed: {0}", code)
           Else
             Console.WriteLine(myId)
           End If
        End Sub




        */








    }
}

Słowa kluczowe: CreateSequentialUUID SequentialUUID

granadaCoder
źródło
0

Mam GitHub Gist z implementacją UUID podobną do Java w C #: https://gist.github.com/rickbeerendonk/13655dd24ec574954366

UUID można utworzyć z najmniejszych i najbardziej znaczących bitów, tak jak w Javie. To także je ujawnia. Implementacja ma jawną konwersję na identyfikator GUID i niejawną konwersję z identyfikatora GUID.

Rick Beerendonk
źródło