如何将外键约束添加到引用两个表B或B的两个表中的表A(ID,类型)?

时间:2019-02-05 15:31:53

标签: sql sql-server azure-sql-database

我希望将表A中的两列用作两个表之一的外键:表B或表C。使用列table_a.item_id和table_a.item_type_id,我想强制任何新行都具有表B或表C中的匹配item_id和item_type_id。

示例:

Table A: Inventory 
+---------+--------------+-------+
| item_id | item_type_id | count |
+---------+--------------+-------+
|       2 |            1 |    32 |
|       3 |            1 |    24 |
|       1 |            2 |    10 |
+---------+--------------+-------+

Table B: Recipes
+----+--------------+-------------------+-------------+----------------------+
| id | item_type_id |       name        | consistency | gram_to_fluid_ounces |
+----+--------------+-------------------+-------------+----------------------+
|  1 |            1 | Delicious Juice   | thin        | .0048472             |
|  2 |            1 | Ok Tasting Juice  | thin        | .0057263             |
|  3 |            1 | Protein Smoothie  | heavy       | .0049847             |
+----+--------------+-------------------+-------------+----------------------+

Table C: Products
+----+--------------+----------+--------+----------+----------+
| id | item_type_id |   name   | price  | in_stock | is_taxed |
+----+--------------+----------+--------+----------+----------+
|  1 |            2 | Purse    | $200   | TRUE     | TRUE     |
|  2 |            2 | Notebook | $14.99 | TRUE     | TRUE     |
|  3 |            2 | Computer | $1,099 | FALSE    | TRUE     |
+----+--------------+----------+--------+----------+----------+

Other Table: Item_Types
+----+-----------+
| id | type_name |
+----+-----------+
|  1 | recipes   |
|  2 | products  |
+----+-----------+

我希望能够拥有一个库存表,员工可以在其中输入库存数量,而不管某项是配方还是产品。我不想有一个product_inventory和recipe_inventory表,因为无论商品类型如何,我都需要对所有库存商品进行很多操作。

一种解决方案是像这样创建参考表:

Table CD: Items
+---------+--------------+------------+-----------+
| item_id | item_type_id | product_id | recipe_id |
+---------+--------------+------------+-----------+
|       2 |            1 | NULL       | 2         |
|       3 |            1 | NULL       | 3         |
|       1 |            2 | 1          | NULL      |
+---------+--------------+------------+-----------+

这似乎很麻烦,而且每当从各自的表中添加/删除产品/食谱时,我现在都需要从此新表中添加/删除产品/食谱。 (是否有自动方法来实现?)

CREATE TABLE [dbo].[inventory] (
    [id] [bigint] IDENTITY(1,1) NOT NULL,
    [item_id] [smallint] NOT NULL,
    [item_type_id] [tinyint] NOT NULL,
    [count] [float] NOT NULL,
CONSTRAINT [PK_inventory_id] PRIMARY KEY CLUSTERED ([id] ASC)
) ON [PRIMARY]

我真正想做的就是这样...

ALTER TABLE [inventory]  
ADD  CONSTRAINT [FK_inventory_sources] FOREIGN KEY ([item_id],[item_type_id])
REFERENCES {[products] ([id],[item_type_id]) OR [recipes] ([id],[item_type_id])}

也许我在描述它时没有解决方案,所以如果您有什么想法可以维护相同/相似的架构,那么我肯定会乐于接受! 谢谢:)

4 个答案:

答案 0 :(得分:2)

我认为您的数据库设计存在缺陷。解决实际问题的最佳方法是将“配方”和“产品”作为一个表。现在,每个表中都有一个名为item_type_id的冗余列。除非您实际上在同一张表中有项目,否则该列将一文不值。我之所以说是多余的,是因为每个表中的每个条目的绝对值都相同。

您有两个选择。如果您无法更改数据库设计,请在没有外键的情况下进行工作,并从正确的表中选择逻辑层。

或者,如果您可以更改数据库设计,则使产品和配方存在于同一表中。您已经有一个item_type表,该表可以标识项的分类,因此将所有项放在同一表中是很有意义的

答案 1 :(得分:1)

您只能为一个或一对列添加一个约束。想想苹果和橙子。一列不能同时引用橙子和苹果。它必须是橙色或苹果。

请注意,这可以通过PERSISTED COMPUTED columns以某种方式实现,但是它只会带来开销和复杂性。

Check This for Reference

答案 2 :(得分:1)

您可以向Inventory表中添加一些计算列:

ALTER TABLE Inventory
    ADD _recipe_item_id AS CASE WHEN item_type_id = 1 THEN item_id END persisted
ALTER TABLE Inventory
    ADD _product_item_id AS CASE WHEN item_type_id = 2 THEN item_id END persisted

然后,您可以使用这两列而不是item_id将两个单独的外键添加到两个表中。我假设这两个表中的item_type_id列已经被适当地计算/约束,但是如果没有,您可能也要考虑这一点。

由于选择了错误的类型时,这些计算列为NULL,并且如果至少一个列值为NULL,则由于SQL Server不会检查FK约束,因此它们都可以存在并且只能存在一个否则任何时候都可以满足。

答案 3 :(得分:1)

由于您的产品和食谱是分开存储的,并且看起来大多具有单独的列,所以单独的库存表可能是正确的方法。例如

CREATE TABLE dbo.ProductInventory
(
        Product_id INT NOT NULL,
        [count] INT NOT NULL,
    CONSTRAINT FK_ProductInventory__Product_id FOREIGN KEY (Product_id) 
        REFERENCES dbo.Product (Product_id)
);

CREATE TABLE dbo.RecipeInventory
(
        Recipe_id INT NOT NULL,
        [count] INT NOT NULL,
    CONSTRAINT FK_RecipeInventory__Recipe_id FOREIGN KEY (Recipe_id) 
        REFERENCES dbo.Recipe (Recipe_id )
);

如果需要将所有类型组合在一起,则可以简单地使用视图:

CREATE VIEW dbo.Inventory
AS
    SELECT  Product_id AS item_id,
            2 AS item_type_id,
            [Count]
    FROM    ProductInventory
    UNION ALL
    SELECT  recipe_id AS item_id,
            1 AS item_type_id
            [Count]
    FROM    RecipeInventory;
GO

如果您创建一个新的item_type,则无论如何都要修改数据库设计以创建一个新表,所以您只需要同时修改视图

另一种可能性是只有一个Items表,然后让Products / Recipes引用此表。因此,您从项目表开始,每个表都有一个唯一的ID:

CREATE TABLE dbo.Items
(
        item_id INT IDENTITY(1, 1) NOT NULL 
        Item_type_id INT NOT NULL,
    CONSTRAINT PK_Items__ItemID PRIMARY KEY (item_id),
    CONSTRAINT FK_Items__Item_Type_ID FOREIGN KEY (Item_Type_ID) REFERENCES Item_Type (Item_Type_ID),
    CONSTRAINT UQ_Items__ItemID_ItemTypeID UNIQUE (Item_ID, Item_type_id)
);

请注意在(item_id, item_type_id)上添加的唯一密钥,这对于以后的引用完整性很重要。

然后您的每个子表与此之间都具有1:1的关系,因此您的产品表将变为:

CREATE TABLE dbo.Products
(
        item_id BIGINT NOT NULL,
        Item_type_id AS 2,
        name VARCHAR(50) NOT NULL,
        Price DECIMAL(10, 4) NOT NULL,
        InStock BIT NOT NULL,
    CONSTRAINT PK_Products__ItemID PRIMARY KEY (item_id),
    CONSTRAINT FK_Products__Item_Type_ID FOREIGN KEY (Item_Type_ID) 
        REFERENCES Item_Type (Item_Type_ID),
    CONSTRAINT FK_Products__ItemID_ItemTypeID FOREIGN KEY (item_id, Item_Type_ID) 
        REFERENCES dbo.Item (item_id, item_type_id)
);

一些注意事项:

  • item_id再次是主键,确保1:1关系。
  • 确保所有item_type_id均设置为2的计算列item_type_id(为2)。这是关键,因为它允许添加外键约束
  • (item_id, item_type_id)上的外键返回到项表。这样可以确保,如果items表中的原始记录的item_type_id为2,则只能将记录插入到product表中。

第三个选项是用于配方和产品的单个表,并使两个都不要求的列都可以为空。 This answer on types of inheritance非常值得一读。