Как создать GUID / UUID в Python


687

Как мне создать GUID в Python, который не зависит от платформы? Я слышал, что есть метод, использующий ActivePython в Windows, но это Windows только потому, что он использует COM. Есть ли метод с использованием простого Python?



35
Ради всего святого, это UUID - универсальный уникальный идентификатор en.wikipedia.org/wiki/Universally_unique_identifier - просто, к сожалению, MS предпочла GUID.
david.barkhuizen

5
Вот один лайнер для вас:python -c 'import uuid; print(uuid.uuid4())'
Ctrl-C

Ответы:


777

Модуль uuid в Python 2.5 и выше обеспечивает RFC-совместимую генерацию UUID. См. Документацию модуля и RFC для деталей. [ источник ]

Docs:

Пример (работает на 2 и 3):

>>> import uuid
>>> uuid.uuid4()
UUID('bd65600d-8669-4903-8a14-af88203add38')
>>> str(uuid.uuid4())
'f50ec0b7-f960-400d-91f0-c42a6d44e3d0'
>>> uuid.uuid4().hex
'9fe2c4e93f654fdbb24c02b15259716c'

20
Кроме того, взгляните на shortuuidмодуль, который я написал, так как он позволяет генерировать более короткие читаемые UUID: github.com/stochastic-technologies/shortuuid
Ставрос Корокитакис,

2
@StavrosKorokithakis: вы случайно написали модуль shortuuid для Python 3.x?
Джей Патель

2
@JayPatel Shortuuid не работает для Python 3? Если нет, пожалуйста, сообщите об ошибке.
Ставрос Корокитакис

1
Какая разница между uuid4().hexа str(uuid4())?
Кевин

6
Ну, как вы можете видеть выше, str(uuid4())возвращает строковое представление UUID с включенными тире, а uuid4().hexвозвращает «UUID в виде шестнадцатеричной строки из 32 символов»
stuartd

324

Если вы используете Python 2.5 или более позднюю версию, модуль uuid уже включен в стандартный дистрибутив Python.

Пример:

>>> import uuid
>>> uuid.uuid4()
UUID('5361a11b-615c-42bf-9bdb-e2c3790ada14')

116

Скопировано из: https://docs.python.org/2/library/uuid.html (поскольку опубликованные ссылки не были активными, и они продолжают обновляться)

>>> import uuid

>>> # make a UUID based on the host ID and current time
>>> uuid.uuid1()
UUID('a8098c1a-f86e-11da-bd1a-00112444be1e')

>>> # make a UUID using an MD5 hash of a namespace UUID and a name
>>> uuid.uuid3(uuid.NAMESPACE_DNS, 'python.org')
UUID('6fa459ea-ee8a-3ca4-894e-db77e160355e')

>>> # make a random UUID
>>> uuid.uuid4()
UUID('16fd2706-8baf-433b-82eb-8c7fada847da')

>>> # make a UUID using a SHA-1 hash of a namespace UUID and a name
>>> uuid.uuid5(uuid.NAMESPACE_DNS, 'python.org')
UUID('886313e1-3b8a-5372-9b90-0c9aee199e5d')

>>> # make a UUID from a string of hex digits (braces and hyphens ignored)
>>> x = uuid.UUID('{00010203-0405-0607-0809-0a0b0c0d0e0f}')

>>> # convert a UUID to a string of hex digits in standard form
>>> str(x)
'00010203-0405-0607-0809-0a0b0c0d0e0f'

>>> # get the raw 16 bytes of the UUID
>>> x.bytes
'\x00\x01\x02\x03\x04\x05\x06\x07\x08\t\n\x0b\x0c\r\x0e\x0f'

>>> # make a UUID from a 16-byte string
>>> uuid.UUID(bytes=x.bytes)
UUID('00010203-0405-0607-0809-0a0b0c0d0e0f')

28

Я использую GUID в качестве случайных ключей для операций типа базы данных.

Шестнадцатеричная форма с черточками и дополнительными символами кажется мне неоправданно длинной. Но мне также нравится, что строки, представляющие шестнадцатеричные числа, очень безопасны, поскольку они не содержат символов, которые могут вызвать проблемы в некоторых ситуациях, таких как «+», «=» и т. Д.

Вместо шестнадцатеричного я использую безопасную для URL строку base64. Следующее не соответствует какой-либо спецификации UUID / GUID, хотя (кроме наличия необходимого количества случайности).

import base64
import uuid

# get a UUID - URL safe, Base64
def get_a_uuid():
    r_uuid = base64.urlsafe_b64encode(uuid.uuid4().bytes)
    return r_uuid.replace('=', '')

2
Если вы не собираетесь использовать его в каких-либо контекстах UUID, вы также можете просто использовать random.getrandbits(128).to_bytes(16, 'little')или (для криптослучайности) os.urandom(16)и получить полные 128 бит случайных (UUIDv4 использует 6-7 бит в информации о версии). Или используйте только 15 байтов (теряя 1-2 бита случайным образом по сравнению с UUIDv4) и избегайте необходимости обрезать =знаки, одновременно уменьшая кодированный размер до 20 байтов (с 24, обрезая до 22), так как любое кратное 3 байта кодирует до #bytes / 3 * 4base64 символов без заполнения.
ShadowRanger

@ ShadowRanger Да, это в основном идея. 128 случайных битов, как можно короче, но и безопасны для URL. В идеале это будет использовать только заглавные и строчные буквы, а затем цифры. Так что я думаю, строка 62-го основания.
Крис

Когда я использую вашу функцию, я получаю ошибку типа из returnоператора, ожидающего байтовоподобный объект. Это можно исправить с помощью return str(r_uuid).replace('=','').
Марк Кортинк

8

Если вам нужно передать UUID для первичного ключа для вашей модели или уникального поля, то приведенный ниже код возвращает объект UUID -

 import uuid
 uuid.uuid4()

Если вам нужно передать UUID в качестве параметра для URL, вы можете сделать, как показано ниже:

import uuid
str(uuid.uuid4())

Если вы хотите шестнадцатеричное значение для UUID, вы можете сделать следующее:

import uuid    
uuid.uuid4().hex

0

Эта функция полностью настраивается и генерирует уникальный идентификатор пользователя на основе указанного формата

например: - [8, 4, 4, 4, 12], этот формат упоминается, и он сгенерирует следующий uuid

LxoYNyXe-7hbQ-caJt-DSdU-PDAht56cMEWi

 import random as r

 def generate_uuid():
        random_string = ''
        random_str_seq = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
        uuid_format = [8, 4, 4, 4, 12]
        for n in uuid_format:
            for i in range(0,n):
                random_string += str(random_str_seq[r.randint(0, len(random_str_seq) - 1)])
            if n != 12:
                random_string += '-'
        return random_string

3
UUID являются стандартными и не переменными по длине. Генерация случайной строки настраиваемым способом может быть полезна в некоторых ситуациях, но не в этом контексте. Вы можете проверить en.wikipedia.org/wiki/Universally_unique_identifier для определения.
miguelr

2
Лучше избегайте этого, иначе вы можете столкнуться с проблемами совместимости (это не стандартные GUID)
Сильвен Гантуа

-1

2019 Ответ (для Windows):

Если вам нужен постоянный UUID, который уникально идентифицирует машину в Windows, вы можете воспользоваться этим трюком: (Скопировано из моего ответа на https://stackoverflow.com/a/58416992/8874388 ).

from typing import Optional
import re
import subprocess
import uuid

def get_windows_uuid() -> Optional[uuid.UUID]:
    try:
        # Ask Windows for the device's permanent UUID. Throws if command missing/fails.
        txt = subprocess.check_output("wmic csproduct get uuid").decode()

        # Attempt to extract the UUID from the command's result.
        match = re.search(r"\bUUID\b[\s\r\n]+([^\s\r\n]+)", txt)
        if match is not None:
            txt = match.group(1)
            if txt is not None:
                # Remove the surrounding whitespace (newlines, space, etc)
                # and useless dashes etc, by only keeping hex (0-9 A-F) chars.
                txt = re.sub(r"[^0-9A-Fa-f]+", "", txt)

                # Ensure we have exactly 32 characters (16 bytes).
                if len(txt) == 32:
                    return uuid.UUID(txt)
    except:
        pass # Silence subprocess exception.

    return None

print(get_windows_uuid())

Использует Windows API для получения постоянного UUID компьютера, затем обрабатывает строку, чтобы убедиться, что это действительный UUID, и, наконец, возвращает объект Python ( https://docs.python.org/3/library/uuid.html ), который дает вам удобный способы использования данных (например, 128-разрядное целое число, шестнадцатеричная строка и т. д.).

Удачи!

PS: вызов подпроцесса, вероятно, можно заменить на ctypes, непосредственно вызывающие ядро ​​/ DLL Windows. Но для моих целей эта функция - все, что мне нужно. Это делает сильную проверку и дает правильные результаты.


-1

Проверьте этот пост, мне очень помог. Короче, лучшим вариантом для меня было:

import random 
import string 

# defining function for random 
# string id with parameter 
def ran_gen(size, chars=string.ascii_uppercase + string.digits): 
    return ''.join(random.choice(chars) for x in range(size)) 

# function call for random string 
# generation with size 8 and string  
print (ran_gen(8, "AEIOSUMA23")) 

Потому что мне нужно было всего 4-6 случайных символов вместо громоздкого GUID.


Это кажется совершенно не связанным с вопросом о UUID.
Сокс с Моникой
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.