ฉันจะสร้าง GUID ใน Python ที่เป็นอิสระจากแพลตฟอร์มได้อย่างไร ฉันได้ยินว่ามีวิธีการใช้ ActivePython บน Windows แต่เป็น Windows เท่านั้นเนื่องจากใช้ COM มีวิธีการใช้ Python ธรรมดาหรือไม่?
python -c 'import uuid; print(uuid.uuid4())'
ฉันจะสร้าง GUID ใน Python ที่เป็นอิสระจากแพลตฟอร์มได้อย่างไร ฉันได้ยินว่ามีวิธีการใช้ ActivePython บน Windows แต่เป็น Windows เท่านั้นเนื่องจากใช้ COM มีวิธีการใช้ Python ธรรมดาหรือไม่?
python -c 'import uuid; print(uuid.uuid4())'
คำตอบ:
โมดูล uuid ใน Python 2.5 ขึ้นไปให้การสร้าง UUID ที่สอดคล้องกับ RFC ดูเอกสารโมดูลและ RFC สำหรับรายละเอียด [ แหล่งที่มา ]
เอกสาร:
ตัวอย่าง (ทำงานกับ 2 และ 3):
>>> import uuid
>>> uuid.uuid4()
UUID('bd65600d-8669-4903-8a14-af88203add38')
>>> str(uuid.uuid4())
'f50ec0b7-f960-400d-91f0-c42a6d44e3d0'
>>> uuid.uuid4().hex
'9fe2c4e93f654fdbb24c02b15259716c'
shortuuid
โมดูลที่ฉันเขียนเพราะจะช่วยให้คุณสร้าง UUID ที่สั้นและอ่านได้: github.com/stochastic-technologies/shortuuid
uuid4().hex
และstr(uuid4())
คืออะไร
str(uuid4())
ส่งคืนการแทนค่าสตริงของ UUID ด้วยเครื่องหมายขีดกลางที่รวมอยู่ในขณะที่uuid4().hex
ส่งกลับ"The UUID เป็นสตริงเลขฐานสิบหก 32 ตัวอักษร"
หากคุณใช้ Python 2.5 หรือใหม่กว่าโมดูล uuidจะรวมอยู่ในการแจกแจงมาตรฐานของ Python แล้ว
Ex:
>>> import uuid
>>> uuid.uuid4()
UUID('5361a11b-615c-42bf-9bdb-e2c3790ada14')
คัดลอกมาจาก: 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')
ฉันใช้ GUID เป็นปุ่มสุ่มสำหรับการดำเนินการประเภทฐานข้อมูล
รูปแบบเลขฐานสิบหกที่มีเครื่องหมายขีดกลางและอักขระพิเศษยาวเกินความจำเป็นสำหรับฉัน แต่ฉันก็ชอบที่สตริงที่ใช้แทนเลขฐานสิบหกนั้นมีความปลอดภัยอย่างมากโดยที่ไม่มีอักขระที่สามารถทำให้เกิดปัญหาในบางสถานการณ์เช่น '+', '=', ฯลฯ
แทนที่จะเป็นเลขฐานสิบหกฉันใช้สตริง base64 ที่ปลอดภัยสำหรับ url ต่อไปนี้ไม่เป็นไปตามข้อกำหนด 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('=', '')
random.getrandbits(128).to_bytes(16, 'little')
หรือ (สำหรับการเข้ารหัสแบบเข้ารหัสลับ) os.urandom(16)
และรับบิตเต็ม 128 บิต (UUIDv4 ใช้ 6-7 บิตกับข้อมูลรุ่น) หรือใช้เพียง 15 ไบต์ (สูญเสียการสุ่ม 1-2 บิตกับ UUIDv4) และหลีกเลี่ยงความต้องการตัด=
สัญญาณขณะเดียวกันก็ลดขนาดการเข้ารหัสลงเหลือ 20 ไบต์ (จาก 24, ลดเหลือ 22) เป็นตัวเข้ารหัส 3 ไบต์ใด ๆ ถึง#bytes / 3 * 4
base64 อักขระโดยไม่ต้องมีการเสริมเต็ม
return
คำสั่งคาดหวังวัตถุคล้ายไบต์ return str(r_uuid).replace('=','')
มันสามารถแก้ไขด้วย
หากคุณต้องการส่งผ่าน UUID สำหรับคีย์หลักสำหรับรุ่นหรือฟิลด์ที่ไม่ซ้ำกันรหัสด้านล่างจะส่งคืนวัตถุ UUID -
import uuid
uuid.uuid4()
หากคุณต้องการผ่าน UUID เป็นพารามิเตอร์สำหรับ URL คุณสามารถทำเช่นรหัสด้านล่าง -
import uuid
str(uuid.uuid4())
หากคุณต้องการค่าฐานสิบหกสำหรับ UUID คุณสามารถทำด้านล่างหนึ่ง -
import uuid
uuid.uuid4().hex
ฟังก์ชันนี้สามารถกำหนดค่าได้อย่างสมบูรณ์และสร้าง uid ที่ไม่ซ้ำกันตามรูปแบบที่ระบุ
เช่น: - [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
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 ที่เรียก Windows kernel / DLLs โดยตรง แต่สำหรับวัตถุประสงค์ของฉันฟังก์ชั่นนี้เป็นสิ่งที่ฉันต้องการ มันตรวจสอบที่แข็งแกร่งและสร้างผลลัพธ์ที่ถูกต้อง
ตรวจสอบโพสต์นี้ช่วยฉันได้มาก ในระยะสั้นตัวเลือกที่ดีที่สุดสำหรับฉันคือ:
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 ขนาดใหญ่