MySQL UUID Smackdown: UUID vs. INT for Primary Key

摘要:本教程向您介绍 MySQL UUID,展示如何使用它作为表的主键 (PK),并讨论使用它作为主键的优缺点。

MySQL UUID简介

UUID 代表通用唯一标识符。 UUID 是根据RFC 4122 “通用唯一标识符 (UUID) URN 命名空间”定义的。

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

在 MySQL 中,UUID 值是一个 128 位数字,表示为由五个十六进制数字组成的 utf8 字符串,格式如下:

aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee
Code language: SQL (Structured Query Language) (sql)

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

UUID()
Code language: SQL (Structured Query Language) (sql)

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

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

mysql> SELECT UUID();
+--------------------------------------+
| uuid()                               |
+--------------------------------------+
| 865234ad-6a92-11e7-8846-b05adad3f0ae |
+--------------------------------------+
1 row in set (0.05 sec)
Code language: SQL (Structured Query Language) (sql)

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 值的大小和未排序,使用 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}
Code language: SQL (Structured Query Language) (sql)

MySQL UUID 示例

我们来看一个使用UUID作为主键的例子。

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

CREATE TABLE customers (
    id BINARY(16) PRIMARY KEY,
    name VARCHAR(255)
);
Code language: SQL (Structured Query Language) (sql)

要将 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');
Code language: SQL (Structured Query Language) (sql)

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

SELECT 
    BIN_TO_UUID(id) id, 
    name
FROM
    customers;   
Code language: SQL (Structured Query Language) (sql)
MySQL UUID Example

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

本教程有帮助吗?