「MySQL」解决中文乱码

MySQL中文乱码是一个常见的问题,解决方法也是非常简单的。由于MySQL默认编码是latin1,所以不支持中文。中文编码有好几种,例如gbkgb2312等等,但是我建议使用utf8编码,并且全局都使用utf8编码。

不过由于现在许多应用都开始Emoji表情,所以我推荐使用兼容Emoji表情的utf8mb4编码。(注意:使用utf8mb4编码要求MySQL的最低版本为5.5.3+

一、关于版本的区别

引用官网的一段说明:

从5.7.18开始不在二进制包中提供my-default.cnf文件。参考:https://dev.mysql.com/doc/refman/5.7/en/binary-installation.html

经过测试,在5.7.18版本中,使用tar.gz安装时,也就是压缩包解压出来安装这种,已经不再需要my.cnf文件也能正常运行。

my.cnf文件就是把在命令行上启动MySQL时后面的参数用cnf文件配置好,那么下载启动时就不再需要在命令上加如参数。

这个my.cnf文件可以是自定义位置,也可以使用如下默认的位置,只要放在默认位置,MySQL自动识别(通过deb或者APT源安装的,初始位置在下方列表):
在Unix和类Unix系统上读取选项文件
/etc/my.cnf : 全局选项
/etc/mysql/my.cnf : 全局选项
SYSCONFDIR/my.cnf : 全局选项
$MYSQL_HOME/my.cnf : 服务器特定选项(仅限服务器)
defaults-extra-file : 指定的文件 --defaults-extra-file(如果有的话)
~/.my.cnf : 用户特定的选项
~/.mylogin.cnf : 用户特定的登录路径选项(仅限客户端)

在上表中,~表示当前用户的主目录的值($HOME)。
首先它会找/etc/my.cnf 这个文件, 如果这个文件不存在,那么它接下来去找/etc/mysql/my.cnf这个文件,依此类推

所有如果你使用的MySQL版本大于或等于5.7.18,就要从上面的路径中查找到my.cnf文件,然后进行配置。
如果是少于5.7.18,直接打开/etc/my.cnf文件进行配置。
对于Windows系统,原理差不多,大家自行谷歌,因为MySQL很少会装在Windows服务器(用windows服务器的基本都是用巨硬全家桶)。

二、解决中文乱码问题

=================================================
MySQL版本:5.7.22
操作系统:Ubuntu 18.04 amd64
=================================================

1、查看MySQL数据库编码

方法一:通过 status 命令

mysql> status
--------------
mysql  Ver 14.14 Distrib 5.7.22, for Linux (x86_64) using  EditLine wrapper

Connection id:      3
Current database:
Current user:       root@localhost
SSL:            Not in use
Current pager:      stdout
Using outfile:      ''
Using delimiter:    ;
Server version:     5.7.22-0ubuntu0.17.10.1 (Ubuntu)
Protocol version:   10
Connection:     Localhost via UNIX socket
Server characterset:    utf8mb4
Db     characterset:    utf8mb4
Client characterset:    utf8mb4
Conn.  characterset:    utf8mb4
UNIX socket:        /var/run/mysqld/mysqld.sock
Uptime:         29 min 55 sec

Threads: 1  Questions: 5  Slow queries: 0  Opens: 107  Flush tables: 1  Open tables: 26  Queries per second avg: 0.002
--------------

看下面这四部分,默认不是utf8

Server characterset:    utf8mb4
Db     characterset:    utf8mb4
Client characterset:    utf8mb4
Conn.  characterset:    utf8mb4

方法二:通过 show variables like '%char%';

mysql> show variables like '%char%';
+--------------------------+----------------------------+
| Variable_name            | Value                      |
+--------------------------+----------------------------+
| character_set_client     | utf8mb4                    |
| character_set_connection | utf8mb4                    |
| character_set_database   | utf8mb4                    |
| character_set_filesystem | binary                     |
| character_set_results    | utf8mb4                    |
| character_set_server     | utf8mb4                    |
| character_set_system     | utf8                       |
| character_sets_dir       | /usr/share/mysql/charsets/ |
+--------------------------+----------------------------+
8 rows in set (0.54 sec)

2、设置MySQL数据库编码

打开 /etc/mysql/my.cnf 文件,修改为如下设置:

[client]  
default-character-set=utf8mb4

[mysql]  
default-character-set=utf8mb4

[mysqld]  
collation-server = utf8mb4_general_ci
character-set-server = utf8mb4  

重启MySQL服务

sudo service mysql restart

或者

/etc/init.d/mysql restart

通过status或者show variables like '%char%';查看编码,已经全部设置为utf8mb4,搞定了。

验证一下

创建数据库 test01

mysql> create database test01;
Query OK, 1 row affected (0.07 sec)

mysql> use test01;
Database changed
mysql> status;
--------------
mysql  Ver 14.14 Distrib 5.7.22, for Linux (x86_64) using  EditLine wrapper

Connection id:      3
Current database:   test01
Current user:       root@localhost
SSL:            Not in use
Current pager:      stdout
Using outfile:      ''
Using delimiter:    ;
Server version:     5.7.22-0ubuntu0.17.10.1 (Ubuntu)
Protocol version:   10
Connection:     Localhost via UNIX socket
Server characterset:    utf8mb4
Db     characterset:    utf8mb4
Client characterset:    utf8mb4
Conn.  characterset:    utf8mb4
UNIX socket:        /var/run/mysqld/mysqld.sock
Uptime:         21 min 47 sec

Threads: 1  Questions: 40  Slow queries: 0  Opens: 120  Flush tables: 1  Open tables: 38  Queries per second avg: 0.030
--------------

创建表t_user

mysql> create table t_user(id int);                                                                                      Query OK, 0 rows affected (0.44 sec)

mysql> show create table t_user;
+--------+---------------------------------------------------------------------------------------------+
| Table  | Create Table                                                                                |
+--------+---------------------------------------------------------------------------------------------+
| t_user | CREATE TABLE `t_user` (
  `id` int(11) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 |
+--------+---------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)

一切正常。

3、解决一些残余问题

可能有些同学发现,我明明已经修改成功了,为什么我通过看status命令查看编码还是latin1,是不是设置出了问题?其实不是,这是因为你进入了某个数据库,然后查看的是当前数据库的编码,这个数据库很可能是你在修改编码前已经创建了,所以我们要自己修改一下数据库编码即可。

3.1、修改数据库编码

查看数据库编码:

mysql> use test
Reading table information for completion of table and column names
You can turn off this feature to get a quicker startup with -A

Database changed
mysql> status
--------------
mysql  Ver 14.14 Distrib 5.7.22, for Linux (x86_64) using  EditLine wrapper

Connection id:      3
Current database:   test
Current user:       root@localhost
SSL:            Not in use
Current pager:      stdout
Using outfile:      ''
Using delimiter:    ;
Server version:     5.7.22-0ubuntu0.17.10.1 (Ubuntu)
Protocol version:   10
Connection:     Localhost via UNIX socket
Server characterset:    utf8mb4
Db     characterset:    latin1
Client characterset:    utf8mb4
Conn.  characterset:    utf8mb4
UNIX socket:        /var/run/mysqld/mysqld.sock
Uptime:         17 sec

Threads: 1  Questions: 24  Slow queries: 0  Opens: 119  Flush tables: 1  Open tables: 38  Queries per second avg: 1.411
--------------

这里的Db characterset:还是latin1

修改数据库编码:

mysql> alter database test CHARACTER SET utf8mb4;
Query OK, 1 row affected (0.05 sec)

mysql> status
--------------
mysql  Ver 14.14 Distrib 5.7.22, for Linux (x86_64) using  EditLine wrapper

Connection id:      3
Current database:   test
Current user:       root@localhost
SSL:            Not in use
Current pager:      stdout
Using outfile:      ''
Using delimiter:    ;
Server version:     5.7.22-0ubuntu0.17.10.1 (Ubuntu)
Protocol version:   10
Connection:     Localhost via UNIX socket
Server characterset:    utf8mb4
Db     characterset:    utf8mb4
Client characterset:    utf8mb4
Conn.  characterset:    utf8mb4
UNIX socket:        /var/run/mysqld/mysqld.sock
Uptime:         7 min 44 sec

Threads: 1  Questions: 29  Slow queries: 0  Opens: 119  Flush tables: 1  Open tables: 38  Queries per second avg: 0.062
--------------

3.2、修改表编码

查看表的编码:

mysql> show create table t_character;
+-------------+-------------------------------------------------------------------------------------------------+
| Table       | Create Table                                                                                    |
+-------------+-------------------------------------------------------------------------------------------------+
| t_character | CREATE TABLE `t_character` (
  `id` int(11) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=latin1 |
+-------------+-------------------------------------------------------------------------------------------------+
1 row in set (0.13 sec)

修改数据表的编码:

mysql> alter table t_character default character set utf8mb4;
Query OK, 0 rows affected (0.63 sec)
Records: 0  Duplicates: 0  Warnings: 0

mysql> show create table t_character;
+-------------+--------------------------------------------------------------------------------------------------+
| Table       | Create Table                                                                                     |
+-------------+--------------------------------------------------------------------------------------------------+
| t_character | CREATE TABLE `t_character` (
  `id` int(11) DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 |
+-------------+--------------------------------------------------------------------------------------------------+
1 row in set (0.00 sec)

参考资料:

  1. MySQL 5.7 my.cnf配置文件说明
  2. MySQL中文编码设置为utf-8
  3. 让应用程序支持emoji字符
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容