MySQL UUID Smackdown:主键的UUID与INT

时间:2019-11-20 08:52:33  来源:igfitidea点击:

简介:本教程向您介绍MySQL UUID,向您展示如何将其用作表的主键(PK),并讨论将其用作主键的优缺点。

MySQL UUID简介

UUID代表通用唯一IDentifier。
UUID是基于RFC 4122("通用唯一标识符(UUID)URN命名空间)"定义的。

UUID被设计为在空间和时间上全局唯一的数字。
即使两个UUID值是在两个独立的服务器上生成的,也应该是不同的。

在MySQL中,UUID值是一个128位数字,以五个格式的utf8字符串表示,格式如下:

aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee

要生成UUID值,请使用UUID()函数,如下所示:

UUID()

UUID()函数返回符合RFC 4122中描述的UUID版本1的UUID值。

例如,以下语句使用UUID()函数生成UUID值:

mysql> SELECT UUID();
+--------------------------------------+
| uuid()                               |
+--------------------------------------+
| 865234ad-6a92-11e7-8846-b05adad3f0ae |
+--------------------------------------+
1 row in set (0.05 sec)

MySQL UUID与自动增量INT作为主键

优点

使用UUID作为主键具有以下优点:

  • UUID值在表,数据库甚至服务器之间都是唯一的,允许您合并来自不同数据库的行或在服务器之间分布数据库。

  • UUID值不会公开有关您的数据的信息,因此可以更安全地在URL中使用。
    例如,如果ID为10的客户通过http://www.example.com/customers/10/ URL访问其帐户,则很容易猜到有一个客户11、12等,这可能是攻击目标。

  • 可以在任何地方生成UUID值,以避免往返数据库服务器。
    它还简化了应用程序中的逻辑。
    例如,要将数据插入父表和子表中,必须先将其插入父表中,获取生成的ID,然后再将数据插入子表中。
    通过使用UUID,您可以预先生成父表的主键值,并在事务中同时将行插入到父表和子表中。

缺点

除了优点之外,UUID值还具有一些缺点:

  • 存储UUID值(16字节)比整数(4字节)甚至是大整数(8字节)需要更多的存储空间。

  • 调试似乎更加困难,想象一下表达式WHERE id ='df3b7cb7-6a95-11e7-8846-b05adad3f0ae'而不是WHERE id = 10

  • 使用UUID值可能会由于其大小而导致性能问题,而不是对其进行排序。

MySQL UUID解决方案

在MySQL中,您可以使用紧凑格式(BINARY)存储UUID值,并借助以下功能以人类可读格式(VARCHAR)显示它们:

  • UUID_TO_BIN
  • BIN_TO_UUID
  • IS_UUID

请注意,UUID_TO_BIN(),BIN_TO_UUID()和IS_UUID()函数仅在MySQL 8.0或更高版本中可用。

UUID_TO_BIN()函数将UUID从人类可读格式(VARCHAR)转换为紧凑格式(BINARY)格式进行存储,而BIN_TO_UUID()函数将UUID从紧凑格式(BINARY)转换为人类可读格式(VARCHAR)用于显示。

如果参数是有效的字符串格式UUID,则IS_UUID()函数将返回1。
如果参数不是有效的字符串格式UUID,则IS_UUID函数将返回0.如果参数为NULL,则IS_UUID()函数将返回NULL。

以下是MySQL中有效的字符串格式UUID:

aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee
aaaaaaaabbbbccccddddeeeeeeeeeeee
{aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee}

MySQL UUID示例

让我们看一个使用UUID作为主键的示例。

以下语句创建一个名为customers的新表:

CREATE TABLE customers (
    id BINARY(16) PRIMARY KEY,
    name VARCHAR(255)
);

要将UUID值插入id列,请使用UUID()和UUID_TO_BIN()函数,如下所示:

INSERT INTO customers(id, name)
VALUES(UUID_TO_BIN(UUID()),'John Doe'),
      (UUID_TO_BIN(UUID()),'Will Smith'),
      (UUID_TO_BIN(UUID()),'Mary Jane');

要从UUID列查询数据,请使用BIN_TO_UUID()函数将二进制格式转换为人类可读格式:

SELECT 
    BIN_TO_UUID(id) id, 
    name
FROM
    customers;

在本教程中,您了解了MySQL UUID以及如何将其用于主键列。