如何使用指向同一表ID的多个键将一对多关系链接起来?

时间:2019-07-17 23:29:45

标签: mysql sql database

我有一个表files和一个表users以及staffclient。 文件表包含分别指向职员,职员和客户的行made_by,行published_by和行about_client

一个用户必须是工作人员或客户。 一个文件可以有多个作者(made_by行),但是只有一个用户可以发布它(published_by),并且只能是一个客户端(about_client)。

首先,我在文件上放了3个外键:

    CONSTRAINT `made_by_fk`    FOREIGN KEY (`made_by`)    REFERENCES `users` (`id`) ON DELETE CASCADE ON UPDATE CASCADE,
    CONSTRAINT `published_by_fk`    FOREIGN KEY (`published_by`)    REFERENCES `users` (`id`) ON DELETE CASCADE ON UPDATE CASCADE,
    CONSTRAINT `about_client_fk`    FOREIGN KEY (`about_client`)    REFERENCES `users` (`id`) ON DELETE CASCADE ON UPDATE CASCADE

但是经过一番研究,我发现我应该将外键放在员工/客户身上,问题就来了,fk是可选的。员工不是文件的作者的强制性文件,否则客户端可能没有任何文件附加到文件中。

我有点迷茫。

DROP TABLE IF EXISTS `files`;
CREATE TABLE IF NOT EXISTS `files` (
    `id`                INT             NOT NULL AUTO_INCREMENT
                                        PRIMARY KEY,
    `made_by`           INT             NOT NULL,
    `published_by`      INT             NOT NULL,
    `about_client`      INT             NOT NULL,
    `creation_date`     DATETIME        NOT NULL,
    `modification_date` DATETIME        NULL 
                                        ON UPDATE CURRENT_TIMESTAMP,
    `path`              VARCHAR(255)    NOT NULL,
    `title`             VARCHAR(100)    NOT NULL UNIQUE,
    `category`          INT             NOT NULL,
    `type`              VARCHAR(30)     NOT NULL,
    `size`              INT             NOT NULL,

    CONSTRAINT `made_by_fk`    FOREIGN KEY (`made_by`)    REFERENCES `users` (`id`) ON DELETE CASCADE ON UPDATE CASCADE,
    CONSTRAINT `published_by_fk`    FOREIGN KEY (`published_by`)    REFERENCES `users` (`id`) ON DELETE CASCADE ON UPDATE CASCADE,
    CONSTRAINT `about_client_fk`    FOREIGN KEY (`about_client`)    REFERENCES `users` (`id`) ON DELETE CASCADE ON UPDATE CASCADE
)
    ENGINE   = InnoDB
    COLLATE  = utf8_unicode_ci;

DROP TABLE IF EXISTS `staff`;
CREATE TABLE IF NOT EXISTS `staff` (
    `id`                INT             NOT NULL AUTO_INCREMENT
                                        PRIMARY KEY,
    `job`               INT             NOT NULL,
    `password`          VARCHAR(255)    NOT NULL
)
    ENGINE   = InnoDB
    COLLATE  = utf8_unicode_ci;

DROP TABLE IF EXISTS `users`;
CREATE TABLE IF NOT EXISTS `users` (
    `id`        INT             NOT NULL AUTO_INCREMENT
                                PRIMARY KEY,
    `name`      VARCHAR(100)    NOT NULL,
    `surname`   VARCHAR(100)    NOT NULL,
    `email`     VARCHAR(50)     NOT NULL UNIQUE
)
ENGINE   =  InnoDB
COLLATE  =  utf8_unicode_ci;

DROP TABLE IF EXISTS `clients`;
CREATE TABLE IF NOT EXISTS `clients` (
    `id`                INT         NOT NULL AUTO_INCREMENT
                        PRIMARY KEY,
    `phone`             VARCHAR(30) NOT NULL,
    `age`               INT         NOT NULL,
    `date_of_birth`     DATE        NOT NULL,
    `security_number`   VARCHAR(99) NOT NULL UNIQUE
)
    ENGINE   =  InnoDB
    COLLATE  =  utf8_unicode_ci;

1 个答案:

答案 0 :(得分:1)

如果可以有多个made_by,则它不能是file表中的一列,因为该列只能包含一个值。

您需要另一个表file_made_by,该表代表文件和作者之间的多对多关系。

CREATE TABLE file_made_by (
    file_id INT NOT NULL,
    author_id INT NOT NULL,
    PRIMARY KEY (file_id, author_id),
    FOREIGN KEY (file_id) REFERENCES files (id) ON DELETE CASCADE ON UPDATE CASCADE,
    FOREIGN KEY (author_id) REFERENCES users (id) ON DELETE CASCADE ON UPDATE CASCADE
);

clientsstaff表中还应包含一个user_id列,该列是users.id的外键。

如果about_client必须是关于客户而不是员工的,那么它应该是clients.id的外键,而不是users.id

如果只允许工作人员为作者,则file_made_by.author_id应该是staff.id的FK,而不是users.id。与files.published_by相同。