Либо ИЛИ ненулевые ограничения в MySQL - PullRequest
11 голосов
/ 04 октября 2008

Каков наилучший способ создать ненулевое ограничение в MySQL, чтобы fieldA и fieldB не могли быть равны NULL. Мне все равно, если одно из них само по себе равно NULL, так как другое поле имеет значение, отличное от NULL. И если они оба имеют ненулевые значения, то это даже лучше.

Ответы [ 5 ]

8 голосов
/ 05 октября 2008

Это не ответ непосредственно на ваш вопрос, а некоторая дополнительная информация.

При работе с несколькими столбцами и проверке, являются ли все значения NULL или один не равен NULL, я обычно использую COALESCE() - это краткое, удобочитаемое и легко поддерживаемое значение, если список увеличивается:

COALESCE(a, b, c, d) IS NULL -- True if all are NULL

COALESCE(a, b, c, d) IS NOT NULL -- True if any one is not null

Это можно использовать в вашем триггере.

7 голосов
/ 04 октября 2008

@ Sklivvz: тестируя с MySQL 5.0.51a, я обнаружил, что он анализирует ограничение CHECK, но не применяет его. Я могу вставить (NULL, NULL) без ошибок. Проверено как MyISAM, так и InnoDB. Впоследствии использование SHOW CREATE TABLE показывает, что ограничение CHECK отсутствует в определении таблицы, даже если при определении таблицы не было ошибки.

Это соответствует руководству MySQL , в котором говорится: «Предложение CHECK анализируется, но игнорируется всеми механизмами хранения.»

Так что для MySQL вам придется использовать триггер, чтобы применить это правило. Единственная проблема заключается в том, что триггеры MySQL не могут ни вызвать ошибку, ни прервать операцию INSERT. Единственное, что вы можете сделать в триггере, чтобы вызвать ошибку, - установить для столбца NOT NULL значение NULL.

CREATE TABLE foo (
  FieldA INT,
  FieldB INT,
  FieldA_or_FieldB TINYINT NOT NULL;
);

DELIMITER //
CREATE TRIGGER FieldABNotNull BEFORE INSERT ON foo
FOR EACH ROW BEGIN
  IF (NEW.FieldA IS NULL AND NEW.FieldB IS NULL) THEN
    SET NEW.FieldA_or_FieldB = NULL;
  ELSE
    SET NEW.FieldA_or_FieldB = 1;
  END IF;
END//

INSERT INTO foo (FieldA, FieldB) VALUES (NULL, 10); -- OK
INSERT INTO foo (FieldA, FieldB) VALUES (10, NULL); -- OK
INSERT INTO foo (FieldA, FieldB) VALUES (NULL, NULL); -- gives error

Вам также необходим аналогичный триггер ДО ОБНОВЛЕНИЯ.

5 голосов
/ 28 февраля 2015

В MySQL 5.5 введено SIGNAL , поэтому нам больше не нужен дополнительный столбец в ответе Билла Карвина. Билл указал, что вам также нужен триггер для обновления, поэтому я тоже включил его.

CREATE TABLE foo (
  FieldA INT,
  FieldB INT
);

DELIMITER //
CREATE TRIGGER InsertFieldABNotNull BEFORE INSERT ON foo
FOR EACH ROW BEGIN
  IF (NEW.FieldA IS NULL AND NEW.FieldB IS NULL) THEN
    SIGNAL SQLSTATE '45000'
    SET MESSAGE_TEXT = '\'FieldA\' and \'FieldB\' cannot both be null';
  END IF;
END//
CREATE TRIGGER UpdateFieldABNotNull BEFORE UPDATE ON foo
FOR EACH ROW BEGIN
  IF (NEW.FieldA IS NULL AND NEW.FieldB IS NULL) THEN
    SIGNAL SQLSTATE '45000'
    SET MESSAGE_TEXT = '\'FieldA\' and \'FieldB\' cannot both be null';
  END IF;
END//
DELIMITER ;

INSERT INTO foo (FieldA, FieldB) VALUES (NULL, 10); -- OK
INSERT INTO foo (FieldA, FieldB) VALUES (10, NULL); -- OK
INSERT INTO foo (FieldA, FieldB) VALUES (NULL, NULL); -- gives error
UPDATE foo SET FieldA = NULL; -- gives error
4 голосов
/ 04 октября 2008

Это стандартный синтаксис для такого ограничения, но MySQL после этого блаженно игнорирует ограничение

ALTER TABLE `generic` 
ADD CONSTRAINT myConstraint 
CHECK (
  `FieldA` IS NOT NULL OR 
  `FieldB` IS NOT NULL
) 
2 голосов
/ 04 октября 2008

Я сделал нечто подобное в SQL Server, я не уверен, будет ли это работать напрямую в MySQL, но:

ALTER TABLE tableName ADD CONSTRAINT constraintName CHECK ( (fieldA IS NOT NULL) OR (fieldB IS NOT NULL) );

По крайней мере, я считаю, что это синтаксис.

Однако помните, что вы не можете создавать проверочные ограничения для таблиц, вы можете проверять только столбцы в одной таблице.

...