本文由horstxu发表php
在mysql中执行show create table <tablename>
指令,能够看到一张表的建表语句,example以下:html
CREATE TABLE `table1` ( `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT, `field1` text COLLATE utf8_unicode_ci NOT NULL COMMENT '字段1', `field2` varchar(128) COLLATE utf8_unicode_ci NOT NULL DEFAULT '' COMMENT '字段2', PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8_unicode_ci; 复制代码
大部分字段咱们都能看懂,可是今天要讨论的是COLLATE关键字。这个值后面对应的utf8_unicode_ci
是什么意思呢?面试的时候用这个题目考一考DBA,应该能够难倒一大部分人。mysql
使用phpmyadmin的开发可能会很是眼熟,由于其中的中文表头已经给出了答案:面试
所谓utf8_unicode_ci
,实际上是用来排序的规则。对于mysql中那些字符类型的列,如VARCHAR
,CHAR
,TEXT
类型的列,都须要有一个COLLATE
类型来告知mysql如何对该列进行排序和比较。简而言之,COLLATE会影响到ORDER BY语句的顺序,会影响到WHERE条件中大于小于号筛选出来的结果,会影响**DISTINCT**
、**GROUP BY**
、**HAVING**
语句的查询结果。另外,mysql建索引的时候,若是索引列是字符类型,也会影响索引建立,只不过这种影响咱们感知不到。总之,凡是涉及到字符类型比较或排序的地方,都会和COLLATE有关。算法
COLLATE
一般是和数据编码(CHARSET
)相关的,通常来讲每种CHARSET
都有多种它所支持的COLLATE
,而且每种CHARSET
都指定一种COLLATE
为默认值。例如Latin1
编码的默认COLLATE
为latin1_swedish_ci
,GBK
编码的默认COLLATE
为gbk_chinese_ci
,utf8mb4
编码的默认值为utf8mb4_general_ci
。sql
这里顺便讲个题外话,mysql中有utf8
和utf8mb4
两种编码,在mysql中请你们忘记**utf8**
,永远使用**utf8mb4**
。这是mysql的一个遗留问题,mysql中的utf8
最多只能支持3bytes长度的字符编码,对于一些须要占据4bytes的文字,mysql的utf8
就不支持了,要使用utf8mb4
才行。markdown
不少COLLATE
都带有_ci
字样,这是Case Insensitive的缩写,即大小写无关,也就是说"A"和"a"在排序和比较的时候是一视同仁的。selection * from table1 where field1="a"
一样能够把field1为"A"的值选出来。与此同时,对于那些_cs
后缀的COLLATE
,则是Case Sensitive,即大小写敏感的。app
在mysql中使用show collation
指令能够查看到mysql所支持的全部COLLATE
。以utf8mb4
为例,该编码所支持的全部COLLATE
以下图所示。oop
图中咱们能看到不少国家的语言本身的排序规则。在国内比较经常使用的是utf8mb4_general_ci
(默认)、utf8mb4_unicode_ci
、utf8mb4_bin
这三个。咱们来探究一下这三个的区别:ui
首先utf8mb4_bin
的比较方法其实就是直接将全部字符看做二进制串,而后从最高位往最低位比对。因此很显然它是区分大小写的。
而utf8mb4_unicode_ci
和utf8mb4_general_ci
对于中文和英文来讲,实际上是没有任何区别的。对于咱们开发的国内使用的系统来讲,随便选哪一个都行。只是对于某些西方国家的字母来讲,utf8mb4_unicode_ci
会比utf8mb4_general_ci
更符合他们的语言习惯一些,general
是mysql一个比较老的标准了。例如,德语字母“ß”
,在utf8mb4_unicode_ci
中是等价于"ss"
两个字母的(这是符合德国人习惯的作法),而在utf8mb4_general_ci
中,它却和字母“s”
等价。不过,这两种编码的那些微小的区别,对于正常的开发来讲,很难感知到。自己咱们也不多直接用文字字段去排序,退一步说,即便这个字母排错了一两个,真的能给系统带来灾难性后果么?从网上找的各类帖子讨论来讲,更多人推荐使用utf8mb4_unicode_ci
,可是对于使用了默认值的系统,也并无很是排斥,并不认为有什么大问题。结论:推荐使用utf8mb4_unicode_ci
,对于已经用了utf8mb4_general_ci
的系统,也没有必要花时间改造。
另外须要注意的一点是,从mysql 8.0开始,mysql默认的CHARSET
已经再也不是Latin1
了,改成了utf8mb4
(参考连接),而且默认的COLLATE也改成了utf8mb4_0900_ai_ci
。utf8mb4_0900_ai_ci
大致上就是unicode
的进一步细分,0900
指代unicode比较算法的编号( Unicode Collation Algorithm version),ai
表示accent insensitive(发音无关),例如e, è, é, ê 和 ë是一视同仁的。相关参考连接1,相关参考连接2
设置COLLATE
能够在示例级别、库级别、表级别、列级别、以及SQL指定。实例级别的COLLATE
设置就是mysql配置文件或启动指令中的collation_connection
系统变量。
库级别设置COLLATE
的语句以下:
CREATE DATABASE <db_name> DEFAULT CHARACTER SET utf8mb4 COLLATE utf8mb4_unicode_ci;
复制代码
若是库级别没有设置CHARSET
和COLLATE
,则库级别默认的CHARSET
和COLLATE
使用实例级别的设置。在mysql8.0如下版本中,你若是什么都不修改,默认的CHARSET
是Latin1
,默认的COLLATE
是latin1_swedish_ci
。从mysql8.0开始,默认的CHARSET
已经改成了utf8mb4
,默认的COLLATE
改成了utf8mb4_0900_ai_ci
。
表级别的COLLATE
设置,则是在CREATE TABLE
的时候加上相关设置语句,例如:
CREATE TABLE (
……
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
复制代码
若是表级别没有设置CHARSET
和COLLATE
,则表级别会继承库级别的CHARSET
与COLLATE
。
列级别的设置,则在CREATE TABLE
中声明列的时候指定,例如
CREATE TABLE ( `field1` VARCHAR(64) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL DEFAULT '', …… ) …… 复制代码
若是列级别没有设置CHARSET
和COLATE
,则列级别会继承表级别的CHARSET
与COLLATE
。
最后,你也能够在写SQL查询的时候显示声明COLLATE
来覆盖任何库表列的COLLATE
设置,不太经常使用,了解便可:
SELECT DISTINCT field1 COLLATE utf8mb4_general_ci FROM table1;
SELECT field1, field2 FROM table1 ORDER BY field1 COLLATE utf8mb4_unicode_ci;
复制代码
若是全都显示设置了,那么优先级顺序是 SQL语句 > 列级别设置 > 表级别设置 > 库级别设置 > 实例级别设置。也就是说列上所指定的COLLATE
能够覆盖表上指定的COLLATE
,表上指定的COLLATE
能够覆盖库级别的COLLATE
。若是没有指定,则继承下一级的设置。即列上面没有指定COLLATE
,则该列的COLLATE
和表上设置的同样。
以上就是关于mysql的COLLATE
相关知识。不过,在系统设计中,咱们仍是要尽可能避免让系统严重依赖中文字段的排序结果,在mysql的查询中也应该尽可能避免使用中文作查询条件。
此文已由做者受权腾讯云+社区发布,更多原文请点击
搜索关注公众号「云加社区」,第一时间获取技术干货,关注后回复1024 送你一份技术课程大礼包!