字符集笔记

1.概念
    所谓字符集,就是按照一定的字符编码方案,对一组特定的符号,分别赋予不同数值编码的集合。Oracle数据库最

早支持的编码方案是US7ASCII。 
2.命名规则  
    <语言><比特位数><编码>
    比如: ZHS16GBK表示采用GBK编码格式、16位(两个字节)简体中文字符集.
3.字符编码方案
3.1.单字节编码:
    单字节7位字符集,可以定义128个字符,最常用的字符集为 US7ASCII.
    单字节8位字符集,可以定义256个字符,适合于欧洲大部分国家   例如:WE8ISO8859P1(西欧、8位、ISO标准

8859P1编码 )
3.2.多字节编码
    变长多字节编码,某些字符用一个字节表示,其它字符用两个或多个字符表示,变长多字节编码常用于对亚洲语言

的支持,例如日语、汉语、印地语等.例如:AL32UTF8(其中AL代表ALL,指适用于所有语言)、 zhs16cgb231280 
    定长多字节编码,每一个字符都使用固定长度字节的编码方案,目前oracle唯一支持的定长多字节编码是AF16UTF16

,也是仅用于国家字符集.
3.3.unicode编码
    Unicode是一个涵盖了目前全世界使用的所有已知字符的单一编码方案,也就是说Unicode为每一个字符提供唯一的

编码。UTF-16是unicode的16位编码方式,是一种定长多字节编码,用2个字节表示一个unicode字符,AF16UTF16是UTF

-16编码字符集。UTF-8 是unicode的8位编码方式,是一种变长多字节编码,这种编码可以用1、2、3个字节表示一个

unicode字符,AL32UTF8,UTF8、UTFE是UTF-8编码字符集.
4.超集与子集
    当一种字符集(字符集A)的编码数值包含所有另一种字符集(字符集B)的编码数值,并且两种字符集相同编码数

值代表相同的字符时,则字符集A是字符集B的超级,或称字符集B是字符集A的子集。Oracle8i 和oracle9i官方文档资

料中备有子集-超级对照表(subset-superset pairs),例如:WE8ISO8859P1是WE8MSWIN1252的子集。由于US7ASCII是

最早的Oracle数据库编码格式,因此有许多字符集是US7ASCII的超集,例如WE8ISO8859P1、ZHS16CGB231280、ZHS16GBK

都是US7ASCII的超集。有关超集子集对照表也可以参照eygle的网

站:http://www.eygle.com/archives/20 ... ubset_superset.html
5.数据库字符集(oracle服务器端字符集)
    数据库字符集在创建数据库时指定,在创建后通常不能更改。在创建数据库时,可以指定字符集(CHARACTER SET)

和国家字符集(NATIONAL CHARACTER SET).以下分别对此两方面阐述.
5.1.字符集
    用来存储CHAR, VARCHAR2, CLOB, LONG等类型数据,用来标示诸如表名、列名以及PL/SQL变量等,用来存储SQL和

PL/SQL程序单元等.
5.2.国家字符集: 
    用于存储NCHAR, NVARCHAR2, NCLOB等类型数据,国家字符集实质上是为oracle选择的附加字符集,主要作用是为了

增强oracle的字符处理能力,因为NCHAR数据类型可以提供对亚洲使用定长多字节编码的支持,而数据库字符集则不能

。国家字符集在oracle9i中进行了重新定义,只能在unicode编码中的AF16UTF16和UTF8中选择,默认值是 AF16UTF16.
5.3.确定数据库字符集
    可以查询以下数据字典或视图查看字符集设置情况,nls_database_parameters,props$,v$nls_parameters.查询结

果中NLS_CHARACTERSET表示字符集,NLS_NCHAR_CHARACTERSET表示国家字符集.例如我的例子:
sys@ORCL> select * from nls_database_parameters;

PARAMETER                 VALUE
------------------------- ----------------------------------------
NLS_LANGUAGE              AMERICAN
NLS_TERRITORY             AMERICA
NLS_CURRENCY              $
NLS_ISO_CURRENCY          AMERICA
NLS_NUMERIC_CHARACTERS    .,
NLS_CHARACTERSET          ZHS16GBK
NLS_CALENDAR              GREGORIAN
NLS_DATE_FORMAT           DD-MON-RR
NLS_DATE_LANGUAGE         AMERICAN
NLS_SORT                  BINARY
NLS_TIME_FORMAT           HH.MI.SSXFF AM
NLS_TIMESTAMP_FORMAT      DD-MON-RR HH.MI.SSXFF AM
NLS_TIME_TZ_FORMAT        HH.MI.SSXFF AM TZR
NLS_TIMESTAMP_TZ_FORMAT   DD-MON-RR HH.MI.SSXFF AM TZR
NLS_DUAL_CURRENCY         $
NLS_COMP                  BINARY
NLS_LENGTH_SEMANTICS      BYTE
NLS_NCHAR_CONV_EXCP       FALSE
NLS_NCHAR_CHARACTERSET    AL16UTF16
NLS_RDBMS_VERSION         9.2.0.7.0

20 rows selected.
5.4.更改数据库字符集
    不得已的时候,需要修改数据库字符集可以通过:Alter database character set ZHS16GBK 来实现,千万不要通过

修改PROPS$的方式修改,应该记住的是新的字符集一定是老字符集的超集.由于更改数据库字符集容易出现数据丢失的情

况.故此方法一定要慎用!
6.客户端字符集
    客户端字符集定义了客户端字符数据的编码方式,任何发自或发往客户端的字符数据均使用客户端定义的字符集编

码,客户端可以看作是能与数据库直接连接的各种应用,客户端字符集是通过设置NLS_LANG参数来设定的.
6.1.NLS_LANG参数格式 
    NLS_LANG=LANGUAGE_TERRITORY.CHARACTERSET
    Language: 显示oracle消息,校验,日期命名 
    Territory :指定默认日期、数字、货币等格式 
    Client character set :指定客户端将使用的字符集 
    例如: NLS_LANG=AMERICAN_AMERICA.US7ASCII 则AMERICAN是语言,AMERICA是地区,US7ASCII是客户端字符集.
6.2.客户端字符集设置方法 
6.2.1.UNIX 环境 
    $NLS_LANG=“simplified chinese”_china.zhs16gbk
    $export NLS_LANG
    编辑oracle用户的profile文件 
6.2.2.Windows 环境 
    编辑注册表 
    HKEY_LOCAL_MACHINE-SOFTWARE-ORACLE-HOME0(9i)
    HKEY_LOCAL_MACHINE-SOFTWARE-ORACLE-KEY_OraDb10g_home1(10g)
7.字符集转换
    最常见到的字符集转换是在EXP/IMP过程中,在一个完整的EMP/IMP过程,可能发生4次字符集转换:
    源数据库字符集.
    Export过程中用户会话字符集 (通过NLS_LANG设定).
    Import过程中用户会话字符集 (通过NLS_LANG设定).
    目标数据库字符集.
    以下简单描述一下可能的字符集转换:
7.1.导出的转换过程 
    在Export过程中,如果源数据库字符集与Export用户会话字符集不一致,会发生字符集转换,并在导出文件的头部

几个字节中存储Export用户会话字符集的ID号。在这个转换过程中可能发生数据的丢失.例如:如果源数据库使用

ZHS16GBK,而Export用户会话字符集使用US7ASCII,由于ZHS16GBK是16位字符集,而US7ASCII是7位字符集,这个转换过

程中,中文字符在US7ASCII中不能够找到对等的字符,所以所有中文字符都会丢失而变成“??”形式,这样转换后生成

的Dmp文件已经发生了数据丢失.因此如果想正确导出源数据库数据,则Export过程中用户会话字符集应等于源数据库字

符集或是源数据库字符集的超集.
7.2.导入的转换过程 
    确定导出数据库字符集环境,通过读取导出文件头,可以获得导出文件的字符集设置 
    确定导入session的字符集,即导入Session使用的NLS_LANG环境变量 
    IMP读取导出文件,读取导出文件字符集ID,和导入进程的NLS_LANG进行比较 
    如果导出文件字符集和导入Session字符集相同,那么在这一步骤内就不需要转换,如果不同,就需要把数据转换

为导入Session使用的字符集。可以看出,导入数据到数据库过程中发生两次字符集转换,第一次,导入文件字符集与导

入Session使用的字符集之间的转换,如果这个转换过程不能正确完成,Import向目标数据库的导入过程也就不能完成.

第二次:导入Session字符集与数据库字符集之间的转换.
7.3.字符集扫描工具
    为了避免在数据库迁移过程中由于字符集不同导致的数据损失,oracle提供了字符集扫描工具(character set 

scanner),通过这个工具我们可以测试在数据迁移过程中由于字符集转换可能带来的问题,然后根据测试结果,确定

数据迁移过程中最佳字符集解决方案.以下简述一下此工具的用法:
    用sysdba用户登陆
    执行$ORACLE_HOME\rdbms\admin\csminst.sql
    然后退回到命令提示符执行
    csscan FULL=Y FROMCHAR=WE8ISO8859P1 TOCHAR=UTF8 log=check.log capture=y array=1000000 process=2
    执行完毕以后查看log文件,可以知道那些转换会失败.
7.4.有用的规则
    字符集的修改或转换是一项比较危险的动作,有可能带来数据的丢失.现实操作中也有一些行之有效的方法来避免,

总结出来就如下2点:
    如果源字符集和目标字符集一致,则保证全部客户端一致.
    如果源和目标不一致,则保证导入和导出客户端一致,且与源或者目标保持一致.
8.OS字符集
8.1.查看OS字符集
    WINDOWS环境:命令行下输入chcp.例如:
    C:\Documents and Settings\zw>chcp
    活动的代码页: 936
    注:代码页936也就是中文字符集 GBK,在Microsoft的官方站点上,我们可以查到关于936代码页的具体编码规则,请

参考以下链接: http://www.microsoft.com/globaldev/reference/dbcs/936.htm
    UNIX环境:SHELL命令env 可以查看系统环境变量,用env |grep LANG 直接得到.例如:
    [oracle@q1test01 ~]$ env |grep LANG
    LANG=en_US.UTF-8
8.2.OS字符集作用
    这里通过一个测试的例子来说明一下OS字符集在字符集转换过程中的作用.
    环境:
    OS:WINXP SP2 中文版,ORACLE:10.2.0.1,服务器和客户端在一台机器上,服务器字符集:AL32UTF8 国家字符集 

AL16UTF16,客户端字符集: AL32UTF8.
    以下是在SQLPLUS里面的操作:
SQL> create table test (a varchar2(10));
琛ㄥ凡鍒涘缓銆?
SQL> insert into test values ('我');
宸插垱寤?1 琛屻?
SQL> commit;
鎻愪氦瀹屾垚銆?
SQL> select a,dump(a) &nbsp;from test;
A&nbsp;
----------&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp; 
DUMP(A)
--------------------------------------------------------------------------------
我&nbsp; 
Typ=96 Len=2: 206,210&nbsp;&nbsp; &nbsp;

    可以看到,服务器端和客户端都是UTF8但是显示出来的提示全部是乱码,再看一下汉字"我"字的编码, 查了一下编码

表,在ZHS16GBK里面是206 98,在UTF8里面是230 136 145,在UTF16里面应该是98 17,但是实际的值是什么呢?实际的编码

值是206 210,既不是GBK,也不是UTF8和UTF16的编码,查了一下其它的编码表,最终发现206 210是GB2312“我”字的编

码.
    接着把数据库客户端编码切换成ZHS16GBK,看看会发生什么
SQL> insert into test values('我');
已创建 1 行。
SQL> select a,dump(a) &nbsp;from test;
A&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;
DUMP(A)&nbsp; 
我&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; &nbsp;&nbsp; 
Typ=96 Len=3: 230,136,145
    可以发现,数据库在插入时,准确的把"我"字转换成UTF8的编码,而在显示的时候又转换成了准确的汉字"我",怎么解

释上面这个现象呢?
    通过试验我们可以发现OS的字符集实际是GB2312的,但是由于客户端字符集和服务器端字符集一致,所以在上传数

据的时候,认为不存在字符集转换,所以就直接把OS的汉字编码上传上去了,这也是我们看到的库中实际存放的是

GB2312“我”字编码 206 210的原因,显示的时候同样的方式,认为数据库服务器与客户端字符集编码一致,不存在字

符集转换,于是把206 210发到客户端,到了客户端,就正好原封不动的把206 210显示成为了汉字“我”。这也是造成

提示乱码的原因,让GBK的窗口去解析UTF8的编码,从而产生乱码.当客户端字符集是GBK的时候,执行INSERT的时候,

ORACLE意识到服务器与客户端字符集不一致,产生一次字符集转换,把“我”字转换成UTF8编码230,136,145存入数据

库中,select的时候做反操作,将UTF8转换成GBK编码显示出来.
8.3.经典"靠"问题
    在一个中文系统中所有的汉字全部显示为"靠"
测试环境:服务器端:ZHS16GBK.客户端:WE8ISO8859P1,OS:GBK.
库里存在汉字“我是从辉”
客户端做select
ORACLE判断服务器客户端字符集不一致,于是做字符集转换,
西欧字符集里面是没有汉字,转换成特殊编码,倒问号,编码为"BF"
客户端得到"BF BF BF BF"
OS是GBK,认为是两个汉字,"BFBF","BFBF"
最终显示结果"靠靠".

全文完!
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值