使用空列创建唯一约束

时间:2011-11-27 21:16:13

标签: sql postgresql database-design null referential-integrity

我有一张包含这种布局的表格:

CREATE TABLE Favorites
(
  FavoriteId uuid NOT NULL PRIMARY KEY,
  UserId uuid NOT NULL,
  RecipeId uuid NOT NULL,
  MenuId uuid
)

我想创建一个类似于此的唯一约束:

ALTER TABLE Favorites
ADD CONSTRAINT Favorites_UniqueFavorite UNIQUE(UserId, MenuId, RecipeId);

但是,这将允许多行具有相同的(UserId, RecipeId),如果MenuId IS NULL。我想允许NULL中的MenuId存储没有关联菜单的收藏夹,但我最多只想要每个用户/配方对中的一行。

到目前为止我的想法是:

  1. 使用一些硬编码的UUID(例如全零)而不是null 但是,MenuId对每个用户的菜单都有一个FK约束,因此我必须为每个用户创建一个特殊的“空”菜单,这很麻烦。

  2. 使用触发器检查是否存在空条目 我认为这是一个麻烦,我喜欢尽可能避免触发器。另外,我不相信他们可以保证我的数据永远不会处于不良状态。

  3. 忘记它并检查中间件或插入函数中是否存在空条目,并且没有此约束。

  4. 我正在使用Postgres 9.0。

    我有什么方法可以忽略吗?

4 个答案:

答案 0 :(得分:316)

创建two partial indexes

CREATE UNIQUE INDEX favo_3col_uni_idx ON favorites (user_id, menu_id, recipe_id)
WHERE menu_id IS NOT NULL;

CREATE UNIQUE INDEX favo_2col_uni_idx ON favorites (user_id, recipe_id)
WHERE menu_id IS NULL;

这样,只有(user_id, recipe_id) menu_id IS NULL的一个组合,有效地实现了所需的约束。

可能的缺点:您不能使用引用(user_id, menu_id, recipe_id)的外键,不能将CLUSTER基于部分索引,而没有匹配WHERE条件的查询不能使用部分索引。 (你似乎不太可能想要一个宽三列的FK参考 - 而是使用PK列。)

如果您需要完整的索引,您也可以从WHERE中删除favo_3col_uni_idx条件,并且您的要求仍然有效。
现在包含整个表格的索引与另一个索引重叠并且变得更大。根据典型查询和NULL值的百分比,这可能有用,也可能没用。在极端情况下,它甚至可能有助于维护所有三个索引(两个部分索引,总数最多)。

除此之外:我建议不要使用mixed case identifiers in PostgreSQL

答案 1 :(得分:57)

您可以在MenuId上创建一个带有合并的唯一索引:

CREATE UNIQUE INDEX
Favorites_UniqueFavorite ON Favorites
(UserId, COALESCE(MenuId, '00000000-0000-0000-0000-000000000000'), RecipeId);

你只需要为COALESCE选择一个永远不会出现在“现实生活”中的UUID。你可能永远不会在现实生活中看到零UUID,但如果你是偏执狂,你可以添加一个CHECK约束(因为他们真的是为​​了让你......):

alter table Favorites
add constraint check
(MenuId <> '00000000-0000-0000-0000-000000000000')

答案 2 :(得分:2)

您可以在单独的表格中存储没有关联菜单的收藏夹:

CREATE TABLE FavoriteWithoutMenu
(
  FavoriteWithoutMenuId uuid NOT NULL, --Primary key
  UserId uuid NOT NULL,
  RecipeId uuid NOT NULL,
  UNIQUE KEY (UserId, RecipeId)
)

答案 3 :(得分:0)

我认为这里存在语义问题。在我看来,用户可以拥有(但仅一个)最喜欢的食谱来准备特定菜单。 (OP有菜单和食谱混合;如果我错了:请在下面交换MenuId和RecipeId) 这意味着{user,menu}应该是此表中的唯一键。它应该指向恰好一个配方。如果用户对此特定菜单没有喜欢的配方此{用户,菜单}密钥对中不存在行。另外:代理键(FaVouRiteId)是多余的:复合主键对关系映射表完全有效。

这会导致表格定义减少:

CREATE TABLE Favorites
( UserId uuid NOT NULL REFERENCES users(id)
, MenuId uuid NOT NULL REFERENCES menus(id)
, RecipeId uuid NOT NULL REFERENCES recipes(id)
, PRIMARY KEY (UserId, MenuId)
);