【Oracle字符集】乱码的产生原理

编辑手记:很多人在数据库各种操作中遭遇过乱码的问题,今天我们分析一下乱码产生的原因。本文节选于《循序渐进Oracle》

通常在现实环境中,存在3个字符集设置:

  • 客户端应用字符集(Client ApplicationCharacter Set);
  • 客户端NLS_LANG参数设置;
  • 服务器端,数据库字符集(Character Set)设置。

由于一个字符在客户端应用(如SQLPLUS、CMD、NOTEPAD等)中以怎样的字符显示取决于客户端操作系统,客户端能够显示怎样的字符,我们就可以在应用中录入这些字符。至于这些字符能否在数据库中正常存储,就和另外的两个字符集设置紧密相关了(通常我们可以忽略应用程序的字符集,这个字符集在应用程序安装时,已经被内在的决定,并且会依据操作系统的相关设置进行选择)。

在传输过程中,客户端NLS_LANG主要用于进行转换判断。如果NLS_LANG等于数据库字符集,则不进行任何转换直接把字符插入数据库;如果不同则进行转换,转换主要有两个任务:

  • 如果存在对应关系,则把相应二进制编码经过映射后(这一步映射以后,所代表的字符可能发生转换)传递给数据库。
  • 如果不存在对应关系,则传递一个替换字符(不同平台的替换字符各不相同,最常见的替换字符是“?”)。

数据库字符集,在和客户端NLS_LANG不同时,会对经过NLS_LANG转换的字符进一步处理:对于?(即不存在对应关系的字符)直接以?形式存放入数据库,对于其他字符,在NLS_LANG和数据库字符集之间进行转换后存入。

下面来看一下最为常见的字符集及乱码的产生。

NLS_LANG字符集与数据库字符集不同

当NLS_LANG字符集与数据库字符集不同,且NLS_LANG不同于客户端字符集设置时,存在以下两种可能。

1、客户端输入的字符在NLS_LANG中没有对应的字符,这时无法转换,NLS_LANG使用替换字符替代这些无法映射的字符(这一步转换在TTS中完成),在很多字符集中这个替代字符就是“?”。

2、当客户端的字符在NLS_LANG中对应了不同的字符时,传递给数据库以后发生转换,存储的是字符,但是已经丢失了元数据,数据库中的字符不再代表客户端的输入。而且这个过程不可逆,这也就是为什么很多时候在客户端输入的是正常的编码,而查询之后会得到未知字符的原因。

下面通过下图来简单说明一下这个过程。

当客户端在WE8ISO8859P15字符集时,输入欧元符号€,这时客户端NLS_LANG和数据库端字符集不同,进行第一次转换,客户端€符号编码是A4,在NLS_LANG转换时,A4对应了NLS_LANG中的“¤”,这一步的转换产生了错误映射。由于数据库字符集不同于NLS_LANG设置,这时进一步的转换发生了,存入数据库的编码变成了C2A4,虽然同NLS_LANG进行了正确的转换,但是客户端录入的数据已经损坏或者丢失了。

可以用我们熟悉的字符集做一个简单的测试(测试环境是客户端代码点对应中文18030字符集,NLS_LANG设置为US7ASCII字符集,数据库CHARACTER SET为ZHS16GBK)。

这时发现,查询出来的是混乱的字符,把这些字符转换为二进制就是:

110010 1100010 1001010 1010100

补全8位就是如下序列:

00110010 01100010 01001010 01010100

我们把首位换成1,得到如下序列:

10110010 11100010 11001010 11010100

接下来看正确的存储格式:

把这个结果转换为二进制表示:

10110010 11100010 11001010 11010100

这个结果正是前面乱码首位补全1后的结果。这个测试说明在US7ASCII转换中文的时候除去了首位的1,这样就丢失了元数据,导致乱码出现,NLS_LANG的转换作用由此可加一斑!

NLS_LANG和数据库字符集相同时

在这种情况下,数据库端对客户端传递过来的编码不进行任何转换(这样可以提高性能),直接存储进入数据库,那么这时候就存在和上面同样的问题,如果客户端传递过来的字符集在数据库中有正确的对应就可以正确存储,如果没有,就会被替换字符置换成,乱码就这样产生了。

如下图所示,当NLS_LANG和数据库字符集设置相同都为UTF8时,客户端的欧元符号的编码A4就不会经过任何转换就插入到数据库中,而在UTF8的数据库中,A4代表的是一个非法字符。

来看一个简单的测试(测试环境是客户端代码点对应中文18030字符集,客户端NLS_LANG为US7ASCII,数据库字符集为US7ASCII)。

我们知道这个时候存入的数据,数据库不进行任何转换,在以下的测试中,看到中文在US7ASCII字符集下得以正确显示。

原文发布于微信公众号 - 数据和云(OraNews)

原文发表时间:2017-01-15

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏源哥的专栏

由于查询语句中日期的格式引起的问题

我这边有一个系统,在一个环境下运行完全正常,但迁到另外一个环境后,其中一个查询功能就莫名其妙的出现了问题,我通过检查,发现有一个很复杂的查询语句,在一个数据库环...

451
来自专栏乐沙弥的世界

MongoDB 单键(列)索引

674
来自专栏容器云生态

shell基本命令

有关文件显示的命令: du --exclude=iso  -sh  .        //统计当前除了iso这个目录的其他文件大小 ls -F        ...

2067
来自专栏吴柯的运维笔记

Linux下常用的shell脚本整理

<转>分享下看到比较好的关于常用的shell脚本,供大家学习: 1、脚本之间互相调用与传递参数   "1.sh"的脚本,接受参数。如下,如果有一个...

3604
来自专栏python3

pymysql--插入300万数据

        (2)使用python协程(遇到I/O操作就切换任务,无需等待--提高效率)

602
来自专栏FreeBuf

源码审计之空指针引用漏洞

*本文原创作者:freezing,本文属FreeBuf原创奖励计划,未经许可禁止转载

953
来自专栏deed博客

十天学会php详细文字教程_入门至精通

1152
来自专栏idba

Redis 删除1.2亿指定前缀的key

因为更换IDC的原因,我们需要迁移缓存到新的机房,开发同学提出老的缓存有1.2亿无效(未设置过期时间)的key和正常在用的业务key,在迁移之前可以先指定前缀将...

481
来自专栏计算机视觉与深度学习基础

Leetcode 85 Maximal Rectangle 推荐!

Given a 2D binary matrix filled with 0's and 1's, find the largest rectangle co...

1945
来自专栏维C果糖

史上最简单的 MySQL 教程(四十)「数据库变量」

系统变量,顾名思义,是系统设置好的变量(皆为全局级别变量),也是用来控制服务器表现的,如autocommit、wait_timeout等。

39112

扫描关注云+社区