gpt4 book ai didi

mysql - 如何建立多个子实体与列表属性之间的关系?

转载 作者:搜寻专家 更新时间:2023-10-30 23:39:35 25 4
gpt4 key购买 nike

我目前正在处理的(基于 PHP 和 MySQL 的)应用程序的模型包含与描述的相似的继承 here .为了这个问题的目的而简化的类结构可以如下所示:

enter image description here

为了将其映射到我使用的数据库 Class Table Inheritance设计模式。这是物理数据模型:

enter image description here

最具体的属性实际上是针对每个子类的。但是有一些属性需要在几个类中使用(但也不是所有类都需要——否则它们可以在 Foo 类/表中进行管理)。当它是一个简单的属性时,它会导致一些代码重复,但这不是一个大问题。但也有一些情况具有复杂的属性。

例如:FooTypeBazFooTypeBuz 应该包含 Whatever 元素的列表。

通常我会使用包含FOREIGN KEY 的表whatever 来实现这种1:n 关系。但在这种情况下,我需要多个 FOREIGN KEYwhatever(对于 foo_type_bazfoo_type_buz,也许还有一些表更多的)。它很脏。

另一种解决方案:类似于表whatever的“外观”表:

enter image description here

看起来更好(对我来说),但我仍然对这个模型不满意。

如何建立多个子实体与一个collection/list属性之间的关系?这个问题有没有优雅的解决方案?也许是最佳实践/设计模式?

最佳答案

记录关系非常简单 - 您可以创建一个表 foo_whatever (foo_id PK, whatever_set_id FK) 并仅为适当的 foo id 插入行。但是,该架构不会对您可以与任何集合关联的子类型强制执行任何约束,但您现有的架构也不会强制子类型相互排斥。可以使用相同的技术同时执行这两项操作。

考虑在所有 foo_* 表中包含一个类型指示符,例如使用 enum('bar', 'baz', 'buz')。这在 foo 中提供了子类型信息(这比连接 3 个表来查找匹配更方便)并允许外键约束和检查约束强制执行排他子类型并限制可以记录在 foo_whatever。是的,它涉及到一些冗余信息,但它很小并且没有更新异常的风险。

使用涉及类型指示符的复合外键约束,以及限制每个子类型表的类型指示符值的检查约束,应该可以解决问题。这是我建议的架构:

CREATE TABLE `foo` (
`id` int(11) NOT NULL AUTO_INCREMENT,
`type` enum('bar','baz','buz') NOT NULL,
PRIMARY KEY (`id`),
KEY `foo_id` (`id`,`type`)
);

CREATE TABLE `foo_type_bar` (
`foo_id` int(11) NOT NULL,
`foo_type` enum('bar','baz','buz') NOT NULL CHECK (foo_type = 'bar'),
PRIMARY KEY (`foo_id`),
KEY `foo_bar_fk` (`foo_id`,`foo_type`),
CONSTRAINT `foo_bar_fk` FOREIGN KEY (`foo_id`, `foo_type`)
REFERENCES `foo` (`id`, `type`) ON DELETE CASCADE ON UPDATE CASCADE
);

CREATE TABLE `foo_type_baz` (
`foo_id` int(11) NOT NULL,
`foo_type` enum('bar','baz','buz') NOT NULL CHECK (foo_type = 'baz'),
PRIMARY KEY (`foo_id`),
KEY `foo_baz_fk` (`foo_id`,`foo_type`),
CONSTRAINT `foo_baz_fk` FOREIGN KEY (`foo_id`, `foo_type`)
REFERENCES `foo` (`id`, `type`) ON DELETE CASCADE ON UPDATE CASCADE
);

CREATE TABLE `foo_type_buz` (
`foo_id` int(11) NOT NULL,
`foo_type` enum('bar','baz','buz') NOT NULL CHECK (foo_type = 'buz'),
PRIMARY KEY (`foo_id`),
KEY `foo_buz_fk` (`foo_id`,`foo_type`),
CONSTRAINT `foo_buz_fk` FOREIGN KEY (`foo_id`, `foo_type`)
REFERENCES `foo` (`id`, `type`) ON DELETE CASCADE ON UPDATE CASCADE
);

CREATE TABLE `foo_whatever` (
`foo_id` int(11) NOT NULL,
`foo_type` enum('bar','baz','buz') NOT NULL CHECK (foo_type IN ('baz', 'buz')),
`whatever_set_id` int(11) NOT NULL,
PRIMARY KEY (`foo_id`),
KEY `whatever_foo_fk` (`foo_id`,`foo_type`),
KEY `whatever_set_fk` (`whatever_set_id`),
CONSTRAINT `whatever_foo_fk` FOREIGN KEY (`foo_id`, `foo_type`)
REFERENCES `foo` (`id`, `type`) ON DELETE CASCADE ON UPDATE CASCADE,
CONSTRAINT `whatever_set_fk` FOREIGN KEY (`whatever_set_id`)
REFERENCES `whatever_set` (`id`) ON DELETE CASCADE ON UPDATE CASCADE
);

但是,由于 MySQL 忽略检查约束,您需要使用触发器来实现相同的目的:

DELIMITER ;;

CREATE TRIGGER foo_bar_insert_type_check
BEFORE INSERT ON foo_type_bar
FOR EACH ROW
BEGIN
IF NEW.foo_type != 'bar' THEN
SIGNAL SQLSTATE '12345'
SET MESSAGE_TEXT = 'Invalid foo_type in foo_type_bar';
END IF;
END;;

CREATE TRIGGER foo_bar_update_type_check
BEFORE UPDATE ON foo_type_bar
FOR EACH ROW
BEGIN
IF NEW.foo_type != 'bar' THEN
SIGNAL SQLSTATE '12345'
SET MESSAGE_TEXT = 'Invalid foo_type in foo_type_bar';
END IF;
END;;

CREATE TRIGGER foo_baz_insert_type_check
BEFORE INSERT ON foo_type_baz
FOR EACH ROW
BEGIN
IF NEW.foo_type != 'baz' THEN
SIGNAL SQLSTATE '12345'
SET MESSAGE_TEXT = 'Invalid foo_type in foo_type_baz';
END IF;
END;;

CREATE TRIGGER foo_baz_update_type_check
BEFORE UPDATE ON foo_type_baz
FOR EACH ROW
BEGIN
IF NEW.foo_type != 'baz' THEN
SIGNAL SQLSTATE '12345'
SET MESSAGE_TEXT = 'Invalid foo_type in foo_type_baz';
END IF;
END;;

CREATE TRIGGER foo_buz_insert_type_check
BEFORE INSERT ON foo_type_buz
FOR EACH ROW
BEGIN
IF NEW.foo_type != 'buz' THEN
SIGNAL SQLSTATE '12345'
SET MESSAGE_TEXT = 'Invalid foo_type in foo_type_buz';
END IF;
END;;

CREATE TRIGGER foo_buz_update_type_check
BEFORE UPDATE ON foo_type_buz
FOR EACH ROW
BEGIN
IF NEW.foo_type != 'buz' THEN
SIGNAL SQLSTATE '12345'
SET MESSAGE_TEXT = 'Invalid foo_type in foo_type_buz';
END IF;
END;;

CREATE TRIGGER foo_whatever_insert_type_check
BEFORE INSERT ON foo_whatever
FOR EACH ROW
BEGIN
IF NEW.foo_type NOT IN ('baz', 'buz') THEN
SIGNAL SQLSTATE '12345'
SET MESSAGE_TEXT = 'Invalid foo_type in foo_whatever';
END IF;
END;;

CREATE TRIGGER foo_whatever_update_type_check
BEFORE UPDATE ON foo_whatever
FOR EACH ROW
BEGIN
IF NEW.foo_type NOT IN ('baz', 'buz') THEN
SIGNAL SQLSTATE '12345'
SET MESSAGE_TEXT = 'Invalid foo_type in foo_whatever';
END IF;
END;;

DELIMITER ;

关于mysql - 如何建立多个子实体与列表属性之间的关系?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36747386/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com