mysql - MariaDB/MySQL 复合唯一索引无效

标签 mysql sql indexing mariadb

情况:我在表上创建了唯一的复合索引,这花了一些时间,没有删除重复的记录,也没有阻止我插入重复的行。

有人知道这里发生了什么吗?

表结构:

> DESCRIBE translations;
+-------------+--------------+------+-----+---------+----------------+
| Field       | Type         | Null | Key | Default | Extra          |
+-------------+--------------+------+-----+---------+----------------+
| id          | int(11)      | NO   | PRI | NULL    | auto_increment |
| text        | text         | NO   |     | NULL    |                |
| language_id | int(11)      | NO   | MUL | NULL    |                |
| parent_id   | int(11)      | YES  | MUL | NULL    |                |
| type        | varchar(255) | YES  | MUL | NULL    |                |
| flag        | varchar(255) | YES  | MUL | NULL    |                |
+-------------+--------------+------+-----+---------+----------------+

索引创建:

> ALTER IGNORE TABLE `translations`
ADD UNIQUE `unique_translations`
(`language_id`, `parent_id`, `type`, `flag`);

Query OK, 12225526 rows affected (4 min 51.91 sec)     
Records: 12225526  Duplicates: 0  Warnings: 0

索引列表:

> SHOW INDEXES FROM `translations`;
+--------------+------------+---------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
| Table        | Non_unique | Key_name            | Seq_in_index | Column_name | Collation | Cardinality | Sub_part | Packed | Null | Index_type | Comment | Index_comment |
+--------------+------------+---------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+
| translations |          0 | PRIMARY             |            1 | id          | A         |    12178547 |     NULL | NULL   |      | BTREE      |         |               |
| translations |          0 | unique_translations |            1 | language_id | A         |        2712 |     NULL | NULL   |      | BTREE      |         |               |
| translations |          0 | unique_translations |            2 | parent_id   | A         |     2435709 |     NULL | NULL   | YES  | BTREE      |         |               |
| translations |          0 | unique_translations |            3 | type        | A         |     2435709 |     NULL | NULL   | YES  | BTREE      |         |               |
| translations |          0 | unique_translations |            4 | flag        | A         |     3044636 |     NULL | NULL   | YES  | BTREE      |         |               |
| translations |          1 | language_id_fk      |            1 | language_id | A         |          26 |     NULL | NULL   |      | BTREE      |         |               |
| translations |          1 | parent_id_fk        |            1 | parent_id   | A         |     1522318 |     NULL | NULL   | YES  | BTREE      |         |               |
| translations |          1 | flag                |            1 | flag        | A         |       10562 |     NULL | NULL   | YES  | BTREE      |         |               |
| translations |          1 | type                |            1 | type        | A         |       30370 |     NULL | NULL   | YES  | BTREE      |         |               |
+--------------+------------+---------------------+--------------+-------------+-----------+-------------+----------+--------+------+------------+---------+---------------+

问题验证:

> SELECT COUNT(`id`) FROM `translations`;
+-------------+
| COUNT(`id`) |
+-------------+
|  12225526   |
+-------------+
1 row in set (3.29 sec)

> SELECT * FROM `translations` ORDER BY `id` DESC LIMIT 1;
+----------+----------------+-------------+-----------+--------------+-------------+
| id       | text           | language_id | parent_id | type         | flag        |
+----------+----------------+-------------+-----------+--------------+-------------+
| 13754252 | text           |          50 |      NULL | text2        | text3       |
+----------+----------------+-------------+-----------+--------------+-------------+
1 row in set (0.01 sec)

> INSERT INTO `translations` VALUES (NULL, "text", 50, NULL, "text2", "text3");
Query OK, 1 row affected (0.00 sec)

> SELECT COUNT(`id`) FROM `translations`;
+-------------+
| COUNT(`id`) |
+-------------+
|  12225527   |
+-------------+
1 row in set (2.19 sec)

机器信息:

root@precise64:~# uname -a
Linux precise64 3.13.0-43-generic #72-Ubuntu SMP Mon Dec 8 19:35:06 UTC 2014 x86_64 x86_64 x86_64 GNU/Linux
root@precise64:~# dpkg -l | grep -i mariadb
ii  libmariadbclient18                  10.0.14+maria-1~precise          amd64        MariaDB database client library
ii  mariadb-client-10.0                 10.0.14+maria-1~precise          amd64        MariaDB database client binaries
ii  mariadb-client-core-10.0            10.0.14+maria-1~precise          amd64        MariaDB database core client binaries
ii  mariadb-common                      10.0.14+maria-1~precise          all          MariaDB database common files (e.g. /etc/mysql/conf.d/mariadb.cnf)
ii  mariadb-server                      10.0.14+maria-1~precise          all          MariaDB database server (metapackage depending on the latest version)
ii  mariadb-server-10.0                 10.0.14+maria-1~precise          amd64        MariaDB database server binaries
ii  mariadb-server-core-10.0            10.0.14+maria-1~precise          amd64        MariaDB database core server files
ii  mysql-common                        10.0.14+maria-1~precise          all          MariaDB database common files (e.g. /etc/mysql/my.cnf)
root@precise64:~# dpkg -l | grep -i mysql  
ii  libdbd-mysql-perl                   4.025-1                          amd64        Perl5 database interface to the MySQL database
ii  libmysqlclient18                    10.0.14+maria-1~precise          amd64        Virtual package to satisfy external depends
ii  mariadb-common                      10.0.14+maria-1~precise          all          MariaDB database common files (e.g. /etc/mysql/conf.d/mariadb.cnf)
ii  mysql-common                        10.0.14+maria-1~precise          all          MariaDB database common files (e.g. /etc/mysql/my.cnf)
ii  php5-mysql                          5.5.9+dfsg-1ubuntu4.5            amd64        MySQL module for php5
rc  php5-mysqlnd                        5.3.10-1ubuntu3.11               amd64        MySQL module for php5 (Native Driver)
ii  phpmyadmin                          4:4.0.10-1                       all          MySQL web administration tool
root@precise64:~#

最佳答案

在以下四列上使用多列唯一索引,以下记录都将被允许并且不会被视为重复:

language_id | parent_id | type         | flag
------------------------------------------------------
         50 |         1 | text2        | text3
         50 |      NULL | text2        | text3
         50 |      NULL | text2        | text3
         50 |         1 | text2        |  NULL

请记住,NULL 表示未知。尽管对 NULL 值进行了索引,但与 NULL 相比的任何值都会返回 NULL。为了使值被视为重复,它们必须通过相等测试,但是:

1 != NULL
NULL != NULL

关于mysql - MariaDB/MySQL 复合唯一索引无效,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27785431/

相关文章:

php - 在单独的类中配置 MySQL 数据库

php - 使用覆盖旧值的新复选框值更新表

mysql - 使用 JdbcTemplate 和 mysql 数据库在 Spring mvc 中更新/删除?

sql - 创建这些表: Could not create constraint or index: Datatype mismatch

mysql 全文搜索不适用于某些关键字

mysql - 如何查询按 2 个属性分组的最大值?

python - SQLAlchemy 按引用关系的混合属性排序

sql - 永远更新查询

c++ - c/c++ 获取添加到排序数组中的元素的索引

mysql - SQL 中总是需要 ID 列吗?