`
oscar999
  • 浏览: 216791 次
  • 来自: ...
文章分类
社区版块
存档分类
最新评论

理解Oracle数据库字符集

 
阅读更多

一.引言

ORACLE 数据库字符集,即Oracle 全球化支持(Globalization Support) ,或即国家语言支持(NLS )其作用是用本国语言和格式来存储、处理和检索数据。利用全球化支持,ORACLE 为用户提供自己熟悉的数据库母语环境,诸如日期格式、数字格式和存储序列等。Oracle 可以支持多种语言及字符集,其中oracle8i 支持48 种语言、76 个国家地域、229 种字符集,而oracle9i 则支持57 种语言、88 个国家地域、235 种字符集。由于oracle 字符集种类多,且在存储、检索、迁移oracle 数据时多个环节与字符集的设置密切相关,因此在实际的应用中,数据库开发和管理人员经常会遇到有关oracle 字符集方面的问题。本文通过以下几个方面阐述,对oracle 字符集做简要分析

二.字符集基本知识

2.1 字符集
实质就是按照一定的字符编码方案,对一组特定的符号,分别赋予不同数值编码的集合。Oracle 数据库最早支持的编码方案是US7ASCII
Oracle
的字符集命名遵循以下命名规则 :
<Language><bit size><encoding>
: < 语言>< 比特位数>< 编码
>
比如: ZHS16GBK 表示采用GBK 编码格式、16 位(两个字节)简体中文字符集

2.2 字符编码方案
<st1:chsdate year="1899" month="12" day="30" islunardate="False" isrocdate="False" w:st="on">2.2.1 </st1:chsdate> 单字节编码
1 )单字节7 位字符集,可以定义128 个字符,最常用的字符集为 US7ASCII
2 )单字节8 位字符集,可以定义256 个字符,适合于欧洲大部分国家

例如:WE8ISO8859P1( 西欧、8 位、ISO 标准8859P1 编码 )
2.2.2
多字节编码

1 )变长多字节编码
某些字符用一个字节表示,其它字符用两个或多个字符表示,变长多字节编码常用于对亚洲语言的支持, 例如日语、汉语、印地语等
例如:AL32UTF8 (其中AL 代表ALL, 指适用于所有语言)、 zhs16cgb231280
2 )定长多字节编码

每一个字符都使用固定长度字节的编码方案,目前oracle 唯一支持的定长多字节编码是AF16UTF16 ,也是仅用于国家字符集
2.2.3 unicode
编码
Unicode
是一个涵盖了目前全世界使用的所有已知字符的单一编码方案,也就是说Unicode 为每一个字符提供唯一的编码。UTF-16unicode16 位编码方式,是一种定长多字节编码,用2 个字节表示一个unicode 字符,AF16UTF16UTF-16 编码字符集。
UTF-8
unicode8 位编码方式,是一种变长多字节编码,这种编码可以用123 个字节表示一个unicode 字符,AL32UTF8UTF8UTFEUTF-8 编码字符集

2.3 字符集超级
当一种字符集(字符集A )的编码数值包含所有另一种字符集(字符集B )的编码数值,并且两种字符集相同编码数值代表相同的字符时,则字符集A 是字符集B 的超级,或称字符集B 是字符集A 的子集。
Oracle8i
oracle9i 官方文档资料中备有子集- 超级对照表(subset-superset pairs ),例如:WE8ISO8859P1WE8MSWIN1252 的子集。由于US7ASCII 是最早的Oracle 数据库编码格式,因此有许多字符集是US7ASCII 的超集,例如WE8ISO8859P1ZHS16CGB231280ZHS16GBK 都是US7ASCII 的超集。

2.4 数据库字符集(oracle 服务器端字符集)
数据库字符集在创建数据库时指定,在创建后通常不能更改。在创建数据库时,可以指定字符集(CHARACTER SET) 和国家字符集(NATIONAL CHARACTER SET)
<st1:chsdate year="1899" month="12" day="30" islunardate="False" isrocdate="False" w:st="on">2.4.1 </st1:chsdate>
字符集
(1)
用来存储CHAR, VARCHAR2, CLOB, LONG 等类型数据
(2)
用来标示诸如表名、列名以及PL/SQL 变量等
(3)
用来存储SQLPL/SQL 程序单元等
2.4.2
国家字符集:
(1)
用以存储NCHAR, NVARCHAR2, NCLOB 等类型数据
(2)
国家字符集实质上是为oracle 选择的附加字符集,主要作用是为了增强oracle 的字符处理能力,因为NCHAR 数据类型可以提供对亚洲使用定长多字节编码的支持,而数据库字符集则不能。国家字符集在oracle9i 中进行了重新定义,只能在unicode 编码中的AF16UTF16UTF8 中选择,默认值是 AF16UTF16
2.4.3
查询字符集参数

可以查询以下数据字典或视图查看字符集设置情况
nls_database_parameters
props$ v$nls_parameters
查询结果中NLS_CHARACTERSET 表示字符集,NLS_NCHAR_CHARACTERSET 表示国家字符集

2.4.4
修改数据库字符集
按照上文所说,数据库字符集在创建后原则上不能更改。如果需要修改字符集,通常需要导出数据库数据,重建数据库,再导入数据库数据的方式来转换,或通过ALTER DATABASE CHARACTER SET 语句修改字符集,但创建数据库后修改字符集是有限制的,只有新的字符集是当前字符集的超集时才能修改数据库字符集,例如UTF8US7ASCII 的超集,修改数据库字符集可使用ALTER DATABASE CHARACTER SET UTF8

2.5 客户端字符集(NLS_LANG 参数)
<st1:chsdate year="1899" month="12" day="30" islunardate="False" isrocdate="False" w:st="on">2.5.1 </st1:chsdate>
客户端字符集含义
客户端字符集定义了客户端字符数据的编码方式,任何发自或发往客户端的字符数据均使用客户端定义的字符集编码, 客户端可以看作是能与数据库直接连接的各种应用,例如sqlplus,exp/imp 等。客户端字符集是通过设置NLS_LANG 参数来设定的。
2.5.2 NLS_LANG
参数格式
NLS_LANG=<language>_<territory>.<client character set>
Language:
显示oracle 消息, 校验,日期命名
Territory
:指定默认日期、数字、货币等格式
Client character set
:指定客户端将使用的字符集
例如: NLS_LANG=AMERICAN_AMERICA.US7ASCII
AMERICAN
是语言,AMERICA 是地区,US7ASCII 是客户端字符集

2.5.3
客户端字符集设置方法
1)UNIX
环境
$NLS_LANG=“simplified chinese”_china.zhs16gbk
$export NLS_LANG
编辑oracle 用户的profile 文件
2)Windows
环境
编辑注册表
Regedit.exe---HKEY_LOCAL_MACHINE---SOFTWARE---ORACLE—HOME0
2.5.4 NLS
参数查询
Oracle
提供若干NLS 参数定制数据库和用户机以适应本地格式,例如有NLS_LANGUAGE,NLS_DATE_FORMAT,NLS_CALENDER 等,可以通过查询以下数据字典或v$ 视图查看。
NLS_DATABASE_PARAMETERS--
显示数据库当前NLS 参数取值,包括数据库字符集取值
NLS_SESSION_PARAMETERS--
显示由NLS_LANG 设置的参数,或经过alter session 改变后的参数值(不包括由NLS_LANG 设置的客户端字符集)
NLS_INSTANCE_PARAMETE--
显示由参数文件init<SID>.ora 定义的参数V$NLS_PARAMETERS-- 显示数据库当前NLS 参数取值
2.5.5
修改NLS 参数
使用下列方法可以修改NLS 参数
1 )修改实例启动时使用的初始化参数文件
2 )修改环境变量 NLS_LANG
3 )使用ALTER SESSION 语句,在oracle 会话中修改

4 )使用某些SQL 函数
NLS
作用优先级别:Sql function>alter session> 环境变量或注册表> 参数文件> 数据库默认参数

三.导入/ 导出与字符集转换

3.1 EXP/IMP
Export Import 是一对读写Oracle 数据的工具。Export Oracle 数据库中的数据输出到操作系统文件中, Import 把这些文件中的数据读到Oracle 数据库中,由于使用exp/imp 进行数据迁移时,数据从源数据库到目标数据库的过程中有四个环节涉及到字符集,如果这四个环节的字符集不一致,将会发生字符集转换。

EXP
____________ _________________ _____________
|imp导入文件|<-|环境变量NLS_LANG|<-|数据库字符集|
------------ ------------------------------

<v:shapetype id="_x0000_t75" coordsize="21600,21600" o:spt="75" o:preferrelative="t" path="m@4@5l@4@11@9@11@9@5xe" filled="f" stroked="f"><v:stroke joinstyle="miter"></v:stroke><v:formulas><v:f eqn="if lineDrawn pixelLineWidth 0"></v:f><v:f eqn="sum @0 1 0"></v:f><v:f eqn="sum 0 0 @1"></v:f><v:f eqn="prod @2 1 2"></v:f><v:f eqn="prod @3 21600 pixelWidth"></v:f><v:f eqn="prod @3 21600 pixelHeight"></v:f><v:f eqn="sum @0 0 1"></v:f><v:f eqn="prod @6 1 2"></v:f><v:f eqn="prod @7 21600 pixelWidth"></v:f><v:f eqn="sum @8 21600 0"></v:f><v:f eqn="prod @7 21600 pixelHeight"></v:f><v:f eqn="sum @10 21600 0"></v:f></v:formulas><v:path o:extrusionok="f" gradientshapeok="t" o:connecttype="rect"></v:path><o:lock v:ext="edit" aspectratio="t"></o:lock></v:shapetype><v:shape id="_x0000_i1025" style="WIDTH: 348pt; HEIGHT: 36.75pt; mso-wrap-style: square; mso-wrap-distance-left: 9.05pt; mso-wrap-distance-right: 9.05pt; mso-position-horizontal-relative: page; mso-position-vertical-relative: page" type="#_x0000_t75"><v:imagedata src="file:///C:/DOCUME~1/lj/LOCALS~1/Temp/msohtml1/01/clip_image001.png" o:href="http://www.csdb.cn/upload/112.bmp" embosscolor="#000002"></v:imagedata><o:lock v:ext="edit" aspectratio="f"></o:lock></v:shape>

IMP
____________ _________________ _____________
|imp导入文件|->|环境变量NLS_LANG|->|数据库字符集|
------------ ------------------------------

<v:shape id="_x0000_i1026" style="WIDTH: 343.5pt; HEIGHT: 36.75pt; mso-wrap-style: square; mso-wrap-distance-left: 9.05pt; mso-wrap-distance-right: 9.05pt; mso-position-horizontal-relative: page; mso-position-vertical-relative: page" type="#_x0000_t75"><v:imagedata src="file:///C:/DOCUME~1/lj/LOCALS~1/Temp/msohtml1/01/clip_image003.png" o:href="http://www.csdb.cn/upload/113.bmp" embosscolor="#000002"></v:imagedata><o:lock v:ext="edit" aspectratio="f"></o:lock> </v:shape>

四个字符集是
1 )源数据库字符集
2Export 过程中用户会话字符集(通过NLS_LANG 设定)
3Import 过程中用户会话字符集(通过NLS_LANG 设定)
4 )目标数据库字符集

3.2 导出的转换过程
Export 过程中,如果源数据库字符集与Export 用户会话字符集不一致,会发生字符集转换,并在导出文件的头部几个字节中存储Export 用户会话字符集的ID 号。在这个转换过程中可能发生数据的丢失。
: 如果源数据库使用ZHS16GBK ,而Export 用户会话字符集使用US7ASCII ,由于ZHS16GBK16 位字符集,US7ASCII7 位字符集,这个转换过程中,中文字符在US7ASCII 中不能够找到对等的字符,所以所有中文字符都会丢失而变成“?? ” 形式,这样转换后生成的Dmp 文件已经发生了数据丢失。
因此如果想正确导出源数据库数据,则Export 过程中用户会话字符集应等于源数据库字符集或是源数据库字符集的超集

3.3 导入的转换过程
1 )确定导出数据库字符集环境
通过读取导出文件头,可以获得导出文件的字符集设置
2 )确定导入session 的字符集,即导入Session 使用的NLS_LANG 环境变量
3IMP 读取导出文件
读取导出文件字符集ID ,和导入进程的NLS_LANG 进行比较
4 )如果导出文件字符集和导入Session 字符集相同,那么在这一步骤内就不需要转换,如果不同,就需要把数据转换为导入Session 使用的字符集。可以看出,导入数据到数据库过程中发生两次字符集转换
第一次: 导入文件字符集与导入Session 使用的字符集之间的转换,如果这个转换过程不能正确完成,Import 向目标数据库的导入过程也就不能完成。
第二次: 导入Session 字符集与数据库字符集之间的转换。
然而,oracle8i 的这种转换只能在单字节字符集之间进行,oracle8i 导入Session 不支持多字节字符集之间的转换,因此为了避免第一次转换,导入Session 使用的NLS_LANG 与导出文件字符集相同,第二次转换(通过SQL*Net )支持任何两种字符集。以上情况在Oracle9i 中略有不同

四.乱码问题

oracle 在数据存储、迁移过程中经常发生字符乱码问题,归根到底是由于字符集使用不当引起。下面以使用客户端sqlplus 向数据库插入数据和导入/ 导出(EXP/IMP )过程为例,说明乱码产生的原因。

4.1 使用客户端sqlplus 向数据库存储数据
这个过程存在3 个字符集设置
1 )客户端应用字符集
2 )客户端NLS_LANG 参数设置
3 )服务器端数据库字符集(Character Set) 设置
客户端应用sqlplus 中能够显示什么样的字符取决于客户端操作系统语言环境( 客户端应用字符集) ,但在应用中录入这些字符后,这些字符能否在数据库中正常存储,还与另外两个字符集设置紧密相关,其中客户端NLS_LANG 参数主要用于字符数据传输过程中的转换判断。常见的乱码大致有两种情形:
1 )汉字变成问号
当从字符集A 转换成字符集B 时,如果转换字符之间不存在对应关系,NLS_LANG 使用替代字符 替代无法映射的字符
2 )汉字变成未知字符(虽然有些是汉字,但与原字符含义不同)
转换存在对应关系,但字符集A 中的字符编码与字符集B 中的字符编码代表不同含义

4.2 发生乱码原因
乱码产生是由于几个字符集之间转换不匹配造成,分以下几种情况:

(注:字符集之间如果不存在子集、超集对应关系时的情况不予考虑,因为这种情况下字符集之间转换必产生乱码)
1
)服务器端数据库字符集与客户端应用字符集相同,与客户端NLS_LANG 参数设置不同

如果客户端NLS_LANG 字符集是其它两种字符集的子集,转换过程将出现乱码。
解决方法:将三种字符集设置成同一字符集,或NLS_LANG 字符集是其它两种字符集的超集
2
)服务器端数据库字符集与客户端NLS_LANG 参数设置相同,与客户端应用字符集不同
如果客户端应用字符集是其它两种字符集的超集时,转换过程将出现乱码,但对于单字节编码存储中文问题,可参看本文第5 章节的分析
3
)客户端应用字符集、客户端NLS_LANG 参数设置、服务器端数据库字符集互不相同
此种情况较为复杂,但三种字符集之间只要有不能转换的字符,则必产生乱码

4.3 导入/ 导出过程出现乱码原因
这个过程存在4 个字符集设置,在3.1 章节中已分析
1 )源数据库字符集
2EXP 过程中NLS_LANG 参数
3IMP 过程中NLS_LANG 参数
4 )目标数据库字符集
出现乱码原因
1
)当源数据库字符集不等于EXP 过程中NLS_LANG 参数,且源数据库字符集是EXP 过程中NLS_LANG 的子集,才能保证导出文件正确,其他情况则导出文件字符乱码
2
EXP 过程中NLS_LANG 字符集不等于IMP 过程中NLS_LANG 字符集,且EXP 过程中NLS_LANG 字符集是IMP 过程中NLS_LANG 字符集的子级, 才能保证第一次转换正常,否则第一次转换中出现乱码。
3
)如果第一次转换正常,IMP 过程中NLS_LANG 字符集是目标数据库字符集的子集或相同,才能保证第二次转换正常,否则则第二次转换中出现乱码

五.单字节编码存储中文问题

由于历史的原因,早期的oracle 没有中文字符集(如oracle6oracle7oracle7.1, 但有的用户从那时起就使用数据库了,并用US7ASCII 字符集存储了中文,或是有的用户在创建数据库时,不考虑清楚,随意选择一个默认的字符集,如WE8ISO8859P1US7ASCII ,而这两个字符集都没有汉字编码,虽然有些时候选用这种字符集好象也能正常使用,但用这种字符集存储汉字信息从原则上说就是错误的,它会给数据库的使用与维护带来一系列的麻烦。
正常情况下,要将汉字存入数据库,数据库字符集必须支持中文,而将数据库字符集设置为US7ASCII 等单字节字符集是不合适的。US7ASCII 字符集只定义了128 个符号,并不支持汉字。另外,如果在SQL*PLUS 中能够输入中文,操作系统缺省应该是支持中文的,但如果在NLS_LANG 中的字符集设置为US7ASCII ,显然也是不正确的,它没有反映客户端的实际情况。但在实际应用中汉字显示却是正确的,这主要是因为Oracle 检查数据库与客户端的字符集设置是同样的,那么数据在客户与数据库之间的存取过程中将不发生任何转换,但是这实际上导致了数据库标识的字符集与实际存入的内容是不相符的。而在SELECT 的过程中,Oracle 同样检查发现数据库与客户端的字符集设置是相同的,所以它也将存入的内容原封不动地传送到客户端,而客户端操作系统识别出这是汉字编码所以能够正确显示。
在这个例子中,数据库与客户端都没有设置成中文字符集,但却能正常显示中文,从应用的角度看好象没问题。然而这里面却存在着极大的隐患,比如在应用lengthsubstr 等字符串函数时,就可能得到意外的结果。
对于早期使用US7ASCII 字符集数据库的数据迁移到oracle8i/9i 中(使用zhs16gbk ),由于原始数据已经按照US7ASCII 格式存储,对于这种情况,可以通过使用Oracle8i 的导出工具,设置导出字符集为US7ASCII ,导出后使用UltraEdit 等工具打开dmp 文件,修改第二、三字符,修改 0001 0354, 这样就可以将US7ASCII 字符集的数据正确导入到ZHS16GBK 的数据库中。

六.结束语

为了避免在数据库迁移过程中由于字符集不同导致的数据损失,oracle 提供了字符集扫描工具(character set scanner ),通过这个工具我们可以测试在数据迁移过程中由于字符集转换可能带来的问题,然后根据测试结果,确定数据迁移过程中最佳字符集解决方案。

分享到:
评论

相关推荐

    理解ORACLE数据库字符集

    【Oracle数据库字符集详解】 Oracle数据库字符集是其全球化支持(Globalization Support)的一部分,旨在为用户提供本地化的数据存储和处理环境。字符集是字符编码的集合,它为每个特定的符号分配一个唯一的数值...

    oracle数据库字符集 无法修改数据库字符集 乱码

    本文将深入探讨Oracle数据库字符集的概念、重要性、修改过程及其潜在问题,特别是当尝试修改数据库字符集时遇到的乱码问题。 ### Oracle数据库字符集概述 Oracle数据库字符集是指用于表示数据库中所有字符数据的...

    Oracle数据库字符集问题解决方案大全.doc

    Oracle 数据库字符集问题解决方案大全 Oracle 数据库字符集问题解决方案大全中,提出了 Oracle 数据库汉字显示异常的解决方案。该文档涵盖了字符集的概念、字符集的设置、字符集的影响因素、解决汉字显示异常的...

    修改Oracle数据库字符集的方法

    ### 修改Oracle数据库字符集的方法 #### 背景与意义 在使用Oracle数据库的过程中,可能会遇到需要更改数据库字符集的情况。这通常发生在原有字符集不能满足新的业务需求时,例如需要支持更多的语言或特殊字符。...

    数据库字符集修改方法

    #### 一、理解Oracle数据库字符集 在深入探讨字符集修改方法之前,我们需要先了解几个基本概念: 1. **字符集**:用于定义如何将二进制数据映射为文本的一种规则集合。 2. **内部使用字符集**:数据库内部用来表示...

    Oracle数据库字符集问题分析及解决方法.pdf

    总之,理解并掌握Oracle数据库字符集的工作原理,及时查询和调整字符集设置,是防止出现乱码问题和数据迁移失败的关键。在实际操作中,应根据具体情况进行细致的规划和配置,确保数据的一致性和完整性。

    oracle英文字符集插入到中文字符集

    Oracle数据库系统支持多种字符集,字符集决定了数据库如何存储和显示字符。英文字符集通常包括ASCII或ISO-8859-1,而中文字符集如GBK、GB2312或UTF-8则能处理中文字符。当一个使用英文字符集的应用程序尝试与使用...

    对Oracle数据库字符集若干问题的探讨.pdf

    Oracle数据库字符集是数据库系统中一个非常重要的概念,它决定了数据如何被编码和解码,影响着数据库中存储和检索的文本数据的正确性。在Oracle数据库中,字符集分为数据库字符集(Database Character Set)和国家...

    Oracle 修改字符集

    ### Oracle 修改字符集 在Oracle数据库管理中,有时我们需要更改...通过以上步骤,我们可以安全地完成Oracle数据库字符集的修改。这不仅可以提高数据库的兼容性和性能,还可以帮助解决由字符集不匹配引起的各种问题。

    Oracle数据库字符集问题分析及解决方法 (1).pdf

    总之,Oracle数据库字符集问题的解决需要对字符集有深入理解,并能够正确查询和配置相关参数。通过合理规划和管理字符集,可以避免因字符集不匹配引发的数据问题,确保数据库系统的稳定运行和数据迁移的成功。

    Oracle数据库字符集转换规律全面剖析

    Oracle数据库字符集转换规律全面剖析是一项关键的议题,尤其对于数据迁移、备份和恢复操作而言。Oracle数据库的字符集决定了如何存储和显示字符,不同字符集间的转换可能导致数据丢失或错误显示。在Export和Import...

    Oracle数据库字符集问题的研究.pdf

    Oracle数据库字符集问题主要涉及到数据库内部和客户端之间的字符编码匹配问题,这直接影响到包含中文字符的数据能否正确显示和处理。字符集的选择和配置是Oracle数据库管理和维护中的关键环节,尤其对于处理多语言...

    Oracle数据库字符集问题总结

    Oracle数据库字符集问题总结主要关注的是在数据迁移和交互时由于字符集差异导致的问题。字符集是决定数据库如何解释和存储字符的规则集合,对于Oracle数据库来说,它直接影响到数据的正确性和兼容性。 首先,Oracle...

    浅析Oracle数据库字符集问题.pdf

    Oracle数据库字符集问题详解 Oracle数据库是全球广泛采用的关系型数据库管理系统,因其高效的数据安全性和完整性控制,以及跨平台的数据兼容性,成为许多企业和组织的核心数据处理工具。然而,在使用Oracle数据库时...

    Oracle数据库字符集问题及其解决方法.pdf

    1. **数据库字符集与客户端字符集不一致**:Oracle数据库系统分为服务器端和客户端。服务器端的字符集决定了数据库中数据的存储格式,而客户端的字符集则影响数据在应用程序中的显示。当两者不匹配时,可能会出现...

    ORACLE数据库字符集[参考].pdf

    Oracle数据库字符集是Oracle全球化支持的核心组成部分,它允许数据库系统以用户本地的语言和格式存储、处理和检索数据。...因此,理解并合理配置Oracle数据库字符集是确保系统全球化和本地化功能正常运行的关键。

    oracle数据库导入导出命令! 更改字符集

    本文将深入探讨Oracle数据库的导入导出命令,特别是如何更改字符集的相关知识。 首先,Oracle数据库提供了两种主要的数据导入导出工具:`expdp`(Export Data Pump)和`impdp`(Import Data Pump)。这两个实用程序...

Global site tag (gtag.js) - Google Analytics