Как я могу генерировать UUID в C #

137

Я создаю файл .idl программно. Как мне создать UUID для интерфейсов и методов Программно.

Могу ли я сгенерировать UUID программно?

Ума Шанкар Субрамани
источник
25
Вы имеете в виду Guid.NewGuid()?
SLaks

Ответы:

222

Вы, наверное, ищете System.Guid.NewGuid().

тюдоровский
источник
33
Вы также можете сделать String UUID = Guid.NewGuid (). ToString ()
Джастин
10
GUID и UUID все одинаковые?
Ума Шанкар Субрамани
17
@ Uma Shankar Subramani: GUID = глобально уникальный идентификатор, UUID = универсально уникальный идентификатор. Разные слова для одной и той же концепции.
Тудор
1
И вам нужно отформатировать GUID в виде строки, отличной от значения по умолчанию, вы можете использовать ToString(string format)перегрузку, которая принимает один из нескольких спецификаторов формата.
Michiel van Oosterhout
7
Вы, вероятно, захотите это сделать, System.Guid.NewGuid().ToString("B").ToUpper()если вы хотите быть совместимыми с некоторыми инструментами MS Build, которые не могут понять UUID в нижнем регистре. Например, vdprojпроекты установки имеют UUID в верхнем регистре и будут выдавать исключение, если вы дадите ему нижний регистр.
Марк Лаката
43

Будьте осторожны: хотя строковые представления для .NET Guid и (RFC4122) UUID идентичны, формат хранения не совпадает . .NET торгует байтами с прямым порядком байтов для первых трех Guidчастей.

Если вы передаете байты (например, как base64), вы не можете просто использовать Guid.ToByteArray()и кодировать его. Вам понадобятся Array.Reverseпервые три части (данные 1-3).

Я делаю это так:

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);

Посмотрите этот ответ для конкретных деталей реализации .NET.

Редактирование : Спасибо Джеффу Уокеру, Code Ranger, за то, что он указал, что внутренние компоненты не имеют отношения к формату байтового массива, который входит и выходит из конструктора байтового массива и ToByteArray().

Бен Мошер
источник
Примечание: я понимаю, что OP, вероятно, имел в виду Guid(так как он предназначен для .idl), но я просто столкнулся с этим. Итак, поехали, Бингеры и Гуглеры.
Бен Мошер
1
Я не могу проверить это, но вы уверены, что вы должны проверить BitConverter.IsLittleEndian, а не просто всегда обращаться. Документы для Guid.ToByteArray () вызывают порядок байтов как младший и говорят, что конструктор совпадает. Спецификация для GUID является прямым порядком байтов. Я думаю, что это должно быть независимо от порядка байтов машины.
Джефф Уокер Код Рейнджер
@JeffWalkerCodeRanger: от Эрика Липперта, давным-давно: blogs.msdn.com/b/ericlippert/archive/2004/05/25/141525.aspx
Бен Мошер,
Я не понимаю, как ссылка Эрика Липперта отвечает на вопрос. Глядя на код Mono по адресу github.com/mono/mono/blob/master/mcs/class/corlib/System/… мне кажется, что они всегда предполагают, что байты находятся в порядке с прямым порядком байтов, независимо от нативного порядка байтов. Это соответствует моему пониманию, что если бы оно зависело от платформы, оно не соответствовало бы семантике MS API или спецификации. Глядя на дизассемблированный mscorelib, можно предположить, что байты в массиве также имеют порядок байтов.
Джефф Уокер Код Рейнджер
Похоже, ты прав. В то время как формат внутреннего хранилища чувствителен к порядку байтов, конструктор ToByteArray- нет; они всегда lil'-endian. Редактировать входящие.
Бен Мошер
3

Вот клиентское решение "последовательного руководства".

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




        */








    }
}

Ключевые слова: CreateSequentialUUID SequentialUUID

granadaCoder
источник
0

У меня есть GitHub Gist с реализацией UUID на Java в стиле C #: https://gist.github.com/rickbeerendonk/13655dd24ec574954366

UUID может быть создан из младших и наиболее значимых битов, как в Java. Это также разоблачает их. Реализация имеет явное преобразование в GUID и неявное преобразование из GUID.

Рик Берендонк
источник