在数据库中,字符乱码属于常见、多发问题。鉴于本人水平顶多只能归于不入流之类,写这篇文章时心里坐卧不安,实在担忧误导你们。内容仅供参考,如有错误,请各位及时指出,我也好学习提升!mysql
MySQL的字符集有4种级别的设置,分别是:服务器级、数据库级、表级、字段级。sql
(1)、能够在my.cnf中设置数据库
[mysqld] default-character-set=gbk (5.1) character-set-server=gbk (5.5)
(2)、能够在启动选项中设置服务器
mysqld --default-character-set=gbk
(3)、能够在编译的时候设置学习
./configure --with-charset=gbk 或 cmake . -default-charset=gbk
若是没有指定服务器的字符集,默认使用latin1为服务器的字符集。spa
(4)、查看当前服务器的字符集server
mysql> show variables like '%char%'; +--------------------------+-----------------------------------------+ | Variable_name | Value | +--------------------------+-----------------------------------------+ | character_set_client | latin1 | | character_set_connection | latin1 | | character_set_database | latin1 | | character_set_filesystem | binary | | character_set_results | latin1 | | character_set_server | latin1 | | character_set_system | utf8 | | character_sets_dir | /usr/local/mysql-5.6.28/share/charsets/ | +--------------------------+-----------------------------------------+
数据库的字符集在建立数据库的时候指定,也能够在建立完数据库以后经过alter database语句修改。若是数据库中已经存在数据,修改数据库字符集并不能将已有的数据按新字符集存放。因此没法经过修改数据库字符集修改数据的内容。对象
设置数据库字符集的规则:blog
(1)、若是指定了字符集和校对规则,则使用指定的规则;string
(2)、若是仅指定字符集而没有指定校对规则,则使用指定的字符集和默认的校对规则;
(3)、若是没有指定字符集和校对规则,则使用服务器的字符集和校对规;
表的字符集是在建表的时候指定的,能够经过alter table语句进行修改。一样,对于表中已经存在的数据,修改字符集不会影响原有的记录,仍将使用原有的字符集。
设置表的字符集的规则同设置数据库的字符集规则。
列的字符集和校对规则能够在建表的时候指定,也能够在修改表的时候调整。(这个不经常使用,仅记录一下)
除了上述的四种字符集外,对实际的应用访问来讲,还存在客户端和服务端之间交互的字符集,以下:
(1)、character_set_client:客户端字符集
(2)、character_set_connection:链接字符集
(3)、character_set_resluts:结果字符集
一般状况下,这3个字符集都应该是相同的,才能保证用户写入的数据被正确的读出,特别是对于中文字符。
set names命令则用于同时修改这3个参数的值。
字符集不一致是致使数据库内中文内容乱码的罪魁祸首。
实验环境:
Server version: 5.6.28 (在此说明实验环境是因为在学习过程当中,从网上参考了部分资料,实验过程与资料描述稍有出入,未查出缘由,只能暂归结为版本不一样所致。)
实验对象:
mysql> show create table char_test\G *************************** 1. row *************************** Table: char_test Create Table: CREATE TABLE `char_test` ( `id` smallint(6) NOT NULL AUTO_INCREMENT, `name` char(20) NOT NULL, PRIMARY KEY (`id`) ) ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=latin1 1 row in set (0.00 sec)
由上可知,char_test表的字符集是latin1,若是不设置正确的字符集,插入中文字符时,必然会出现以下错误:
mysql> insert into char_test (name) values ('小王'); ERROR 1366 (HY000): Incorrect string value: '\xE5\xB0\x8F\xE7\x8E\x8B' for column 'name' at row 1
解决方案
(1)、先set names latin1,而后再插入数据。
mysql> set names latin1; Query OK, 0 rows affected (0.00 sec) mysql> insert into char_test (name) values ('小王'); Query OK, 1 row affected (0.01 sec) mysql> select * from char_test; +----+--------+ | id | name | +----+--------+ | 1 | Tom | | 2 | 小明 | | 3 | 小王 | +----+--------+ 3 rows in set (0.00 sec)
(2)、在data.sql文件中指定set names latin1,而后经过source命令导入data.sql。
# vi data.sql set names latin1; insert into char_test (name) values ('小李'); mysql> source data.sql Query OK, 1 row affected (0.00 sec) mysql> select * from char_test; +----+--------+ | id | name | +----+--------+ | 1 | Tom | | 2 | 小明 | | 3 | 小王 | | 4 | 小李 | +----+--------+ 4 rows in set (0.00 sec)
(3)、在data.sql文件中指定set names latin1,而后经过mysql命令导入
# vi data.sql set names latin1; insert into char_test (name) values ('小张'); # mysql -uroot -p test1 < data.sql
# mysql -uroot -p -e "set names latin1;select * from test1.char_test;"
(4)、经过指定mysql命令的字符集参数实现 --default-charset-set=字符集
# vi data.sql insert into char_test (name) values ('小张'); # 错误方法 # mysql -uroot -p test1 < data.sql Enter password: ****** ERROR 1366 (HY000) at line 1: Incorrect string value: '\xE5\xB0\x8F\xE8\xB5\xB5' for column 'name' at row 1 # 正确方法 # mysql -uroot -p --default-character-set=latin1 test1 < data.sql Enter password: ****** # mysql -uroot -p -e "set names latin1;select * from test1.char_test;" Enter password: ****** +----+--------+ | id | name | +----+--------+ | 1 | Tom | | 2 | 小明 | | 3 | 小王 | | 4 | 小李 | | 5 | 小张 | | 6 | 小赵 | +----+--------+
(5)、在配置文件中指定客户端的字符集
vi my.cnf [client] default-character-set=latin1 mysql> insert into char_test (name) values ('小马'); Query OK, 1 row affected (0.00 sec) mysql> select * from char_test; +----+--------+ | id | name | +----+--------+ | 1 | Tom | | 2 | 小明 | | 3 | 小王 | | 4 | 小李 | | 5 | 小张 | | 6 | 小赵 | | 7 | 小马 | +----+--------+ 7 rows in set (0.00 sec)