79

是否可以使用UUID值作为 SQLite 中的主键?我发现有关该主题的信息极为有限,因此我不确定 SQLite 是否支持 UUID 数据类型。我应该将 UUID 存储为字符串吗?

4

5 回答 5

49

SQLite 允许使用任何数据类型作为主键。

UUID 可以存储为字符串(人类可读)或 16 字节的 BLOB(如果记录太小以至于差异很重要,这可能会更快)。

于 2013-06-24T14:34:52.113 回答
49

CL 的答案是正确的,但有点绕开了手头的问题。如前所述,任何类型的一列(或多列)都可以用作主键。因此,您可以将 UUID 存储为格式化的、人类可读的字符串格式,并将其作为您的表的键。由于 UUID 只是一个 128 位整数,因此您还可以将整数的字节存储为 BLOB,我想这会稍微快一些。

但是更直接地回答我认为是手头的问题,不,SQLite 没有任何直接支持 UUID 的功能。当 SQLite 创建表时,它使用列的声明类型来确定它将使用五个底层存储类(整数、实数、文本、blob 或 null)中的哪一个。创建表后,不使用列的声明类型。因此没有特定于 UUID 的列类型或存储类。似乎也没有任何函数可用于在格式化的 UUID 字符串之间进行转换。要获取 UUID 的字节,您需要查看编写应用程序的语言提供了哪些方法。例如,Java 的UUID类或 Apple 的

于 2014-06-04T16:55:08.703 回答
11

现在有一个 sqlite 扩展,可以根据https://sqlite.org/src/file/ext/misc/uuid.c创建有效的 uuid

于 2020-12-21T21:52:32.843 回答
5

不确定是否将其用作默认字段,但如果有人需要在 sqlite 查询中生成唯一值,可以使用以下建议的方法:

randomblob(N) 函数返回一个包含伪随机字节的 N 字节 blob。如果 N 小于 1,则返回一个 1 字节的随机 blob。提示:应用程序可以使用此函数与 hex() 和/或 lower() 一起生成全局唯一标识符,如下所示:

hex(randomblob(16)) 

或者

lower(hex(randomblob(16))) 
于 2015-11-19T11:59:33.733 回答
5

我需要在 sqlite 中实现 UUID,因为它不是本机功能,所以这是我在互联网上遇到的一个技巧。SQLite 不支持 UUID,因此想法是创建一个函数,该函数将使用该randomblob()函数生成 UUID

select lower(hex( randomblob(4)) || '-' || hex( randomblob(2))
         || '-' || '4' || substr( hex( randomblob(2)), 2) || '-'
         || substr('AB89', 1 + (abs(random()) % 4) , 1)  ||
         substr(hex(randomblob(2)), 2) || '-' || hex(randomblob(6))) 

这将确保您将拥有一个可以存储在表中的 UUID varchar,因此现在来实现它。SQLite 不存储函数,因此您可以使用在表中插入新记录后可以调用的触发器

CREATE TABLE UUID_TABLE(
   id varchar(500),
   name varchar(500) NOT NULL,
   CONSTRAINT name_unique UNIQUE (name),
   CONSTRAINT rid_pkey PRIMARY KEY (id)
);

和触发器

CREATE TRIGGER AutoGenerateGUID_RELATION_3
AFTER INSERT ON UUID_TABLE
FOR EACH ROW
WHEN (NEW.relation_id IS NULL)
BEGIN
   UPDATE UUID_TABLE SET relation_id = (select lower(hex( randomblob(4)) || '-' ||      hex( randomblob(2))
             || '-' || '4' || substr( hex( randomblob(2)), 2) || '-'
             || substr('AB89', 1 + (abs(random()) % 4) , 1)  ||
             substr(hex(randomblob(2)), 2) || '-' || hex(randomblob(6))) ) WHERE rowid = NEW.rowid;
END;

因此,每当插入新行时,默认情况下 NULL 值将影响 id,之后触发器会将其修改为存储为 varchar 的新 UUID 值。

解决方案灵感来自:解决方案来源

于 2020-04-02T20:39:15.273 回答