2

目前我们有一个关于 MySQL 在枚举字段中的排序顺序的有趣问题。字段枚举条目已按我们想要的顺序排序。为了节省起见,我们在它周围添加了一个 CONCAT,因此它将被强制转换为 char 并按字母顺序排序,正如 MySQL-reference ( MySQL Reference - Enum )所建议的那样

通过编码 ORDER BY CAST(col AS CHAR) 或 ORDER BY CONCAT(col) 确保该列是按词法而不是按索引号排序的。

但这并没有产生预期的结果,所以我们开始进一步调查。似乎 order by 语句不适用于 enum 和 concat 函数的组合。我编写了以下示例脚本,它应该说明我的观点:

CREATE TABLE test (
  `col1` enum('a','b','c') COLLATE utf8_bin DEFAULT NULL
) ENGINE=MyISAM DEFAULT CHARSET=utf8 COLLATE=utf8_bin;

INSERT INTO test
VALUES ('b'), ('c'), ('a');

SELECT * FROM test; -- b, c, a
SELECT * FROM test ORDER BY col1 ASC; -- a, b, c
SELECT * FROM test ORDER BY CAST(col1 AS CHAR) ASC; -- a, b, c
SELECT * FROM test ORDER BY CAST(col1 AS BINARY) ASC; -- a, b, c
SELECT * FROM test ORDER BY CONCAT(col1) ASC; -- b, c, a - This goes wrong

我目前怀疑整理/编码存在某种问题,但我不确定。我的数据库默认编码也是 utf8。MySQL 版本是 5.6.12,但它似乎可以用 MySQL 5.1 重现。存储引擎是 MyIsam,但它也出现在内存引擎中。

任何帮助,将不胜感激。

更新:

看来问题仅在 MySQL 5.6 和列的排序规则中产生。使用第一个 CREATE TABLE 语句,查询工作正常。

CREATE TABLE test (
  `col1` enum('a','b','c') COLLATE utf8_general_ci DEFAULT NULL
)

第二个他们没有。

CREATE TABLE test (
  `col1` enum('a','b','c') COLLATE utf8_bin DEFAULT NULL
)

表和/或数据库的排序规则似乎不会影响查询。可以在此SQL Fiddle中测试查询

4

2 回答 2

2

奇怪,它在这个小提琴中起作用。你有触发器还是什么?

http://sqlfiddle.com/#!2/0976a/2

但是,在 5.6 中出现了问题:

http://sqlfiddle.com/#!9/0976a/1

可能是Mysql错误。

此外,如果您以“正确”的顺序输入枚举中的值,它会起作用:

http://sqlfiddle.com/#!9/a3784/1

在文档中:

ENUM 值根据它们的索引号进行排序,这取决于枚举成员在列规范中列出的顺序。例如,对于 ENUM('b', 'a'),'b' 排在 'a' 之前。

于 2013-09-25T14:27:57.230 回答
0

根据文件

在该Handling of Enumeration Literals部分下,它指出:

如果将数字存储到 ENUM 列中,则该数字被视为可能值的索引,存储的值是具有该索引的枚举成员。(但是,这不适用于 LOAD DATA,它将所有输入都视为字符串。)如果引用了数值,则如果枚举值列表中没有匹配的字符串,它仍然被解释为索引。由于这些原因,不建议使用看起来像数字的枚举值来定义 ENUM 列,因为这很容易让人混淆。例如,以下列的枚举成员的字符串值为“0”、“1”和“2”,但数字索引值为 1、2 和 3:

 numbers ENUM('0','1','2')

如果存储 2,则将其解释为索引值,并变为“1”(索引为 2 的值)。如果存储“2”,则它与枚举值匹配,因此存储为“2”。如果存储'3',它不匹配任何枚举值,所以它被视为一个索引,并成为'2'(索引为3的值)。

mysql> INSERT INTO t (numbers) VALUES(2),('2'),('3');
mysql> SELECT * FROM t;
+---------+
| numbers |
+---------+
| 1       |
| 2       |
| 2       |
+---------+

在你的情况下:

INSERT INTO test
VALUES ('2'), ('3'), ('1');

'2' 的索引值为 2,'3' 为 3,'1' 为 1。所以输出为2,3,1

于 2013-09-25T14:47:34.193 回答