2

我是 MySQL 和 SQL 新手,我刚刚发现 MySQL 不支持断言。

我有这些表:


create table stagione
  (
    nome           varchar(20), 
    biennio        char(9), 
    teatro         varchar(20),
    primary key(nome, biennio),
    foreign key (teatro) references teatro(nome) 
on update cascade on delete set null
  )
  ENGINE=InnoDB;

create table produzione
  (
    produttore     varchar(20), 
    spettacolo     varchar(40),
    primary key(produttore, spettacolo),
    foreign key (produttore) references produttore(nome)
                on update cascade on delete cascade,
    foreign key (spettacolo) references spettacolo(titolo)
                on update cascade on delete cascade
) ENGINE=InnoDB;

create table proposta
  (
   nomeStagione varchar(20),
   biennioStagione char(9),
   spettacolo varchar(40),
   primary key(nomeStagione, biennioStagione, spettacolo),
   foreign key (nomeStagione, biennioStagione) references stagione(nome, biennio)
on update cascade on delete cascade,
   foreign key (spettacolo) references spettacolo(titolo)
 on update cascade on delete cascade
  )
ENGINE=InnoDB;

使用 mysql 我无法编写断言。是否可以使用一个或多个触发器来模拟以下断言?


crate assertion RA2 check (
    not exists (
      select stagione.teatro, stagione.nome, stagione.biennio, count(*)
      from (stagione join proposta on 
           (stagione.nome = proposta.nomeStagione) and 
           (stagione.biennio = proposta.biennioStagione)) 
           join produzione on 
           (proposta.spettacolo = produzione.spettacolo) and
           (stagione.teatro = produzione.produttore)
      group by stagione.teatro, stagione.nome, stagione.biennio
      having count(*) > 2
    )
  );          

我应该怎么写那个触发器?

4

1 回答 1

4

如果我正确理解您的架构,那么您可以通过这样的触发器强制执行您的规则

CREATE TRIGGER tg_proposta_before_insert
BEFORE INSERT ON proposta
FOR EACH ROW
  SET NEW.nomeStagione = IF(
    (
      SELECT COUNT(*) total
        FROM proposta p JOIN produzione d
          ON p.spettacolo = d.spettacolo
       WHERE p.nomeStagione = NEW.nomeStagione
         AND p.biennioStagione = NEW.biennioStagione
         AND d.produttore = 
             (
               SELECT produttore
                 FROM produzione
                WHERE spettacolo = NEW.spettacolo
                LIMIT 1
             )
    ) < 2,
    NEW.nomeStagione,
    NULL
  );

为了让它工作,你必须对table中的列有NOT NULL约束nomeStagioneproposta无论如何,您最好对所有参与主键的列都有这样的约束

一些解释:

  1. 这是一个BEFORE触发器,因为它是 MySql 中唯一可以更改插入行的列值的事件。NEW您可以使用关键字访问这些值。
  2. 现在,当您尝试在其中插入一行时,请proposta检查您尝试插入的同一戏剧是否已经有两部戏剧违反了对列的NOT NULL约束(它可以是任何其他列),从而有效地阻止了此插入完成.nomeStagione

这是SQLFiddle演示。尝试取消注释最后一个插入语句和Build Schema. 您会看到它不允许您插入该行。这将是该剧院本赛季的第三场演出

于 2013-07-28T06:21:21.533 回答