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)。
首先,我使用的版本:
$ 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)。