MySQL: запрос с двумя отношениями многие ко многим и дубликатами, с полными данными из подзапросов - PullRequest
2 голосов
/ 12 июля 2020

Этот вопрос расширяет вопрос, который я только что задал, и здесь он был только что решен: { ссылка }

Но есть необходимость в новом вопросе, так как отсутствует одна деталь. в другом. Но обо всем по порядку.

У меня четыре модели: articles, authors и tags. У каждой статьи может быть много авторов, а также может быть много тегов.

Итак, моя БД будет иметь следующие таблицы:

article
article_author
author
article_tag
tags

Здесь MySQL:

DROP TABLE IF EXISTS article_tag;
DROP TABLE IF EXISTS article_author;
DROP TABLE IF EXISTS author;
DROP TABLE IF EXISTS tag;
DROP TABLE IF EXISTS article;

CREATE TABLE IF NOT EXISTS author (
  id INT(11) NOT NULL AUTO_INCREMENT,
  name VARCHAR(255),
  PRIMARY KEY (id)
);

CREATE TABLE IF NOT EXISTS article (
  id INT(11) NOT NULL AUTO_INCREMENT,
  title VARCHAR(255),
  PRIMARY KEY (id)
);

CREATE TABLE IF NOT EXISTS tag (
  id INT(11) NOT NULL AUTO_INCREMENT,
  tag VARCHAR(255),
  PRIMARY KEY (id)
);

CREATE TABLE IF NOT EXISTS article_author (
  article_id INT(11) NOT NULL,
  author_id INT(11) NOT NULL,
  PRIMARY KEY (article_id, author_id),
  INDEX fk_article_author_article_idx (article_id ASC) VISIBLE,
  INDEX fk_article_author_author_idx (author_id ASC) VISIBLE,
  CONSTRAINT fk_article_author_article
    FOREIGN KEY (article_id)
    REFERENCES article (id),
  CONSTRAINT fk_article_author_author
    FOREIGN KEY (author_id)
    REFERENCES author (id)
);

CREATE TABLE IF NOT EXISTS article_tag (
  article_id INT(11) NOT NULL,
  tag_id INT(11) NOT NULL,
  PRIMARY KEY (article_id, tag_id),
  INDEX fk_article_tag_article_idx (article_id ASC) VISIBLE,
  INDEX fk_article_tag_tag_idx (tag_id ASC) VISIBLE,
  CONSTRAINT fk_article_tag_article
    FOREIGN KEY (article_id)
    REFERENCES article (id),
  CONSTRAINT fk_article_tag_tag
    FOREIGN KEY (tag_id)
    REFERENCES tag (id)
);

И мы можем вставить некоторые данные в нашу базу данных:

INSERT INTO article (id, title) VALUES (1, 'first article'), (2, 'second article'), (3, 'third article');
INSERT INTO author (id, name) VALUES (1, 'first author'), (2, 'second author'), (3, 'third author'), (4, 'fourth author');
INSERT INTO tag (id, tag) VALUES (1, 'first tag'), (2, 'second tag'), (3, 'third tag'), (4, 'fourth tag'), (5, 'fifth tag');
INSERT INTO article_tag (article_id, tag_id) VALUES (1, 1), (1, 2), (1, 3), (2, 2), (2, 4), (2, 5), (3, 1), (3, 2);
INSERT INTO article_author (article_id, author_id) VALUES (1, 1), (1, 2), (1, 3), (2, 2), (2, 4), (3, 1), (3, 2), (3, 3), (3, 4);

Итак, в предыдущем вопросе я извлекал статьи, и для каждой статьи я извлекал связанные идентификаторы авторов, а также идентификаторы тегов. Предлагаемое решение действительно элегантно:

SELECT a.id, a.title,
  (SELECT JSON_ARRAYAGG(aa.author_id)
    FROM article_author aa
    WHERE a.id = aa.article_id
  ) as authors,
    (SELECT JSON_ARRAYAGG(art.tag_id)
    FROM article_tag art
    WHERE a.id = art.article_id
  ) as tags
FROM article a;

Но теперь я хочу получить идентификаторы авторов, а также их имена; то же самое с тегами: идентификатор тега и сам тег.

По-видимому, это нетривиальный вопрос, но я хотел бы знать, какие у меня есть варианты для этого.

1 Ответ

1 голос
/ 12 июля 2020

Вы можете получить нужные данные, используя JOIN s в подзапросе. Если вам нужны только имена, вы можете использовать:

SELECT a.id, a.title,
       (SELECT JSON_ARRAYAGG(au.name)
        FROM article_author aa JOIN
             author au
             ON au.id = aa.author_id
        WHERE a.id = aa.article_id
       ) as authors,
       (SELECT JSON_ARRAYAGG(t.tag)
        FROM article_tag art JOIN
             tag t
             ON art.tag_id = t.id
        WHERE a.id = art.article_id
       ) as tags
FROM article a;

Я не уверен, какую структуру данных вы хотите использовать как с идентификаторами, так и с именами. Если вам нужен массив из JSON объектов с двумя полями в каждом объекте:

SELECT a.id, a.title,
       (SELECT JSON_ARRAYAGG(JSON_OBJECT('name', au.name, 'id', au.id))
        FROM article_author aa JOIN
             author au
             ON au.id = aa.author_id
        WHERE a.id = aa.article_id
       ) as authors,
       (SELECT JSON_ARRAYAGG(JSON_OBJECT('tag', t.tag, 'id', t.id))
        FROM article_tag art JOIN
             tag t
             ON art.tag_id = t.id
        WHERE a.id = art.article_id
       ) as tags
FROM article a;

Здесь - скрипт db <> для этой версии.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...