MariaDB PCRE 给出错误 1644。我错过了什么?

MariaDB PCRE gives ERROR 1644. What am I missing?

首先,我使用的版本:

$ mysql --version
mysql  Ver 15.1 Distrib 10.0.29-MariaDB, for debian-linux-gnu (x86_64) using readline 5.2

除了最后的陈述之外,以下所有内容都可以正常工作:

CREATE DATABASE db1; USE db1;
CREATE TABLE tb1 (id INT NOT NULL AUTO_INCREMENT PRIMARY KEY,
    word VARCHAR(50) NOT NULL);
INSERT INTO tb1 (word) VALUES ('foo');
DELETE FROM tb1 WHERE word='foo';
DELIMITER $$
CREATE TRIGGER before_word_insert BEFORE INSERT ON tb1 FOR EACH ROW
    BEGIN
        SET NEW.word=TRIM(NEW.word);
        IF (NEW.word REGEXP "^\w+$" = 0) THEN
            SIGNAL SQLSTATE '12345' SET MESSAGE_TEXT = 'Invalid word';
        END IF;
    END$$
DELIMITER ;
INSERT INTO tb1 (word) VALUES ('foo');

这导致:

ERROR 1644 (12345): Invalid word

这令人费解,因为 Perl 兼容的正则表达式 \w+ 肯定应该匹配字符串 foo。因此,我原以为最终的 SQL 语句会在没有错误的情况下执行。

那么,是什么导致了错误?

您应该使用单引号。所以你写了这个:

NEW.word REGEXP '^\w+$' = 0

它被解析为:

(NEW.word REGEXP '^\w+$') = 0

我认为这就是您出错的原因。因为当正则表达式匹配的时候,那么括号里的部分returns "1",测试不通过。不匹配时为returns“0”,即通过测试。 = 0 实际上是 NOT.

(注:=作比较,与REGEXP根据documentation具有相同的优先级。)

您不想要也不需要 =:

NEW.word REGEXP '^\w+$'

我认为这里的问题很简单,就是您没有按照 Here

所述转义反斜杠或使用 SQL 模式

Note, the backslash characters (here, and in all examples in the sections below) must be escaped with another backslash, unless you're using the SQL_MODE NO_BACKSLASH_ESCAPES.

This example tests if a character has hex code 0x61: SELECT 'a' RLIKE '\x{61}';

您当前的代码对乱码字符串 '\w' 的计算结果为 1

并非所有正则表达式都有效。切换到此(使用 '")。

REGEXP '^[[:alnum:]]+$'

(这个答案适用于 MariaDB、Percona 和 Oracle 的 MySQL)。