数据库创建以后,如果需要修改
字符集,通常需要重建数据库,通过导入导出的方式来转换。
我们也可以通过以下方式更改
ALTER DATABASE CHARACTER SET
|
注意:修改
数据库字符集时必须谨慎,修改
之前一定要为数据库备份。由于不能回退这项操作,因此可能会造成数据丢失或者损坏。
这是最简单的转换字符集的方式,但并不总是有效。
这个命令在Oracle8时被引入Oracle,这个操作在本质上并不转换任何数据库字符,只是简单的更新数据库中所有跟字符集相关的信息。
这意味着,你只能在新字符集是旧字符集严格超集的情况下使用这种方式转换。
所谓超集是指:
当前字符集中的每一个字符在新字符集中都可以表示,并使用同样的代码点
比如很多字符集都是US7ASCII的严格超集。
如果不是超集,将获得以下错误:
SQL> ALTER DATABASE CHARACTER SET ZHS16CGB231280;
ALTER DATABASE CHARACTER SET ZHS16CGB231280
*
ERROR at line 1:
ORA-12712: new character set must be a superset of old character set
|
下面我们来看一个测试(以下测试在Oracle9.2.0下进行,Oracle9i较Oracle8i在编码方面有较大改变,在Oracle8i中,测试结果可能略有不同):
SQL> select name,value$ from props$ where name like '%NLS%';
NAME VALUE$
------------------------------ ------------------------------
NLS_LANGUAGE AMERICAN
NLS_TERRITORY AMERICA
NLS_CURRENCY $
NLS_ISO_CURRENCY AMERICA
NLS_NUMERIC_CHARACTERS .,
NLS_CHARACTERSET
US7ASCII
NLS_CALENDAR GREGORIAN
NLS_DATE_FORMAT DD-MON-RR
NLS_DATE_LANGUAGE AMERICAN
……………….
NLS_NCHAR_CHARACTERSET AL16UTF16
NLS_RDBMS_VERSION 9.2.0.4.0
20 rows selected.
SQL> select name,dump(name) from eygle.test;
NAME DUMP(NAME)
------------------------------------------------------
测试 Typ=1 Len=4: 178,226,202,212
Test Typ=1 Len=4: 116,101,115,116
2 rows selected.
|
转换字符集,数据库应该在RESTRICTED模式下进行.
c:\>sqlplus "/ as sysdba"
SQL*Plus: Release 9.2.0.4.0 - Production on Sat Nov 1 10:52:30 2003
Copyright (c) 1982, 2002, Oracle Corporation. All rights reserved.
Connected to:
Oracle9i Enterprise Edition Release 9.2.0.4.0 - Production
With the Partitioning, Oracle Label Security, OLAP and Oracle Data Mining options
JServer Release 9.2.0.4.0 - Production
SQL> shutdown immediate
Database closed.
Database dismounted.
ORACLE instance shut down.
SQL> STARTUP MOUNT;
ORACLE instance started.
Total System Global Area 76619308 bytes
Fixed Size 454188 bytes
Variable Size 58720256 bytes
Database Buffers 16777216 bytes
Redo Buffers 667648 bytes
Database mounted.
SQL> ALTER SESSION SET SQL_TRACE=TRUE;
Session altered.
SQL> ALTER SYSTEM ENABLE RESTRICTED SESSION;
System altered.
SQL> ALTER SYSTEM SET JOB_QUEUE_PROCESSES=0;
System altered.
SQL> ALTER SYSTEM SET AQ_TM_PROCESSES=0;
System altered.
SQL> ALTER DATABASE OPEN;
Database altered.
SQL> set linesize 120
SQL> ALTER DATABASE CHARACTER SET ZHS16GBK;
ALTER DATABASE CHARACTER SET ZHS16GBK
*
ERROR at line 1:
ORA-12721: operation cannot execute when other sessions are active
SQL> ALTER DATABASE CHARACTER SET ZHS16GBK;
ALTER DATABASE CHARACTER SET ZHS16GBK
*
ERROR at line 1:
ORA-12716: Cannot ALTER DATABASE CHARACTER SET when CLOB data exists
在Oracle9i中,如果数据库存在CLOB类型字段,那么就不允许对字符集进行转换
SQL>
|
这时候,我们可以去查看alert<sid>.log日志文件,看CLOB字段存在于哪些表上:
ALTER DATABASE CHARACTER SET ZHS16GBK
SYS.METASTYLESHEET (STYLESHEET) - CLOB populated
ORA-12716 signalled during: ALTER DATABASE CHARACTER SET ZHS16GBK...
|
对于不同情况,Oracle提供不同的解决方案,如果是用户数据表,一般我们可以把包含CLOB字段的表导出,然后drop掉相关对象,
转换后再导入数据库;对于系统表,可以按照以下方式处理:
SQL> truncate table Metastylesheet;
Table truncated.
|
然后可以继续进行转换!
SQL> ALTER SESSION SET SQL_TRACE=TRUE;
Session altered.
SQL> ALTER DATABASE CHARACTER SET ZHS16GBK;
Database altered.
SQL> ALTER SESSION SET SQL_TRACE=FALSE;
Session altered.
|
在9.2.0中,转换完成以后,可以通过运行catmet.sql脚本来重建Metastylesheet表:
SQL> @?/rdbms/admin/catmet.sql
|
转换后的数据:
SQL> select name,value$ from props$ where name like '%NLS%';
NAME VALUE$
------------------------------ ------------------------------
NLS_LANGUAGE AMERICAN
NLS_TERRITORY AMERICA
NLS_CURRENCY $
NLS_ISO_CURRENCY AMERICA
NLS_NUMERIC_CHARACTERS .,
NLS_CHARACTERSET
ZHS16GBK
…..
NLS_NCHAR_CHARACTERSET AL16UTF16
NLS_RDBMS_VERSION 9.2.0.4.0
20 rows selected.
SQL> select * from eygle.test;
NAME
------------------------------
测试
test
2 rows selected.
|
提示:
通过设置sql_trace,我们可以跟踪很多数据库的后台操作,这个工具是DBA常用的“利器”之一。
我们简单看一下数据库更改字符集时的后台处理,我提取了主要的更新部分。
通过以下跟踪过程,我们看到数据库在更改字符集的时候,主要更新了12张数据字典表,修改
了数据库的原数据,这也证实了我们以前的说法:
这个更改字符集的操作在本质上并不转换任何数据库字符,只是简单的更新数据库中所有跟字符集相关的信息。
update col$ set charsetid = :1
where
charsetform = :2
update argument$ set charsetid = :1
where
charsetform = :2
update collection$ set charsetid = :1
where
charsetform = :2
update attribute$ set charsetid = :1
where
charsetform = :2
update parameter$ set charsetid = :1
where
charsetform = :2
update result$ set charsetid = :1
where
charsetform = :2
update partcol$ set spare1 = :1
where
charsetform = :2
update subpartcol$ set spare1 = :1
where
charsetform = :2
update props$ set value$ = :1
where
name = :2
update "SYS"."KOTAD$" set SYS_NC_ROWINFO$ = :1
where
SYS_NC_OID$ = :2
update seq$ set increment$=:2,minvalue=:3,maxvalue=:4,cycle#=:5,order$=:6,
cache=:7,highwater=:8,audit$=:9,flags=:10
where
obj#=:1
update kopm$ set metadata = :1, length = :2
where
name='DB_FDO'
|
在这里我们顺便纠正一个由来以及的错误方法.
经常可以在网上看到这样的更改字符集的方法:
1)用SYS用户名登陆ORACLE。
2)查看字符集内容
SQL>SELECT * FROM PROPS$;
3)修改
字符集
SQL> update props$ set value$='新字符集' where name='NLS_CHARACTERSET
'
4) COMMIT;
|
我们看到很多人在这个问题上遇到了惨痛的教训,使用这种方式更改字符集,如果你的value$值输入了不正确的字符集,在8i中那么你
的数据库可能会无法启动,这种情况是非常严重的,有时候你必须从备份中进行恢复;如果是在9i中,可以重新启动数据库后再修改
回正
确的字符集。但是我们仍然不建议使用这种方式进行任何数据库修改
,这是一种极其危险的操作。
实际上当我们更新了字符集,数据库启动时会根据数据库的字符集自动的来修改
控制文件的字符集,如果字符集可以识别,更新控制文
件字符集等于数据库字符集;如果字符集不可识别,那么控制文件字符集更新为US7ASCII.
通过更新props$表的方式修改
字符集,在Oracle7之后就不应该被使用.
以下是我的测试结果,但是严禁一切不备份的修改
研究,即使是对测试库的。
SQL> update props$ set value$='EYGLE' where name='NLS_CHARACTERSET
';
1 row updated.
SQL> commit;
Commit complete.
SQL> select name,value$ from props$ where name like '%NLS%';
NAME VALUE$
------------------------------ -----------------------------------
NLS_LANGUAGE AMERICAN
NLS_TERRITORY AMERICA
NLS_CURRENCY $
NLS_ISO_CURRENCY AMERICA
NLS_NUMERIC_CHARACTERS .,
NLS_CHARACTERSET
EYGLE
NLS_CALENDAR GREGORIAN
NLS_DATE_FORMAT DD-MON-RR
NLS_DATE_LANGUAGE AMERICAN
….
NLS_NCHAR_CHARACTERSET ZHS16GBK
NLS_RDBMS_VERSION 8.1.7.1.1
18 rows selected.
重新启动数据库,发现alert.log文件中记录如下操作:
Mon Nov 03 16:11:35 2003
Updating character set in controlfile to US7ASCII
Completed: ALTER DATABASE OPEN
启动数据库后恢复字符集设置:
SQL> update props$ set value$='ZHS16GBK' where name='NLS_CHARACTERSET
';
1 row updated.
SQL> commit;
Commit complete.
SQL> select name,value$ from props$ where name like '%NLS%';
NAME 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_COMP BINARY
NLS_NCHAR_CHARACTERSET ZHS16GBK
NLS_RDBMS_VERSION 8.1.7.1.1
18 rows selected.
重新启动数据库后,发现控制文件的字符集被更新:
Mon Nov 03 16:21:41 2003
Updating character set in controlfile to ZHS16GBK
Completed: ALTER DATABASE OPEN
|
理解了字符集调整的内部操作以后,我们可以轻易的指出,以上的方法是不正确的,通过前面 ” ALTER DATABASE CHARACTER SET” 方式更改字
符集时,Oracle至少需要更改12张数据字典表,而这种直接更新props$表的方式只完成了其中十二分之一的工作,潜在的完整性隐患是可想而知的。
所以,更改字符集尽量要使用正常的途径
。
分享到:
相关推荐
4. **修改数据导入/导出**:如果你正在导入或导出数据,并遇到字符集问题,可能需要在操作中指定正确的字符集,比如在使用`SQL*Loader`或`EXPDP/IMPDP`工具时。 5. **jar包更新**:"nls_charset12.jar"可能包含了...
### 修改Oracle字符集 #### 适用版本与注意事项 本文档主要介绍了如何在Oracle 9i、10g以及XE版本中修改字符集的方法,并且在10.2.0.1.0版本下进行了成功的测试。需要注意的是,在执行字符集转换的过程中可能会...
- 修改`NLS_LANG`键值为所需的字符集格式,例如:`NLS_LANG=AMERICAN_AMERICA.UTF8`。这里的`AMERICAN_AMERICA`表示语言和地区,而`UTF8`则表示字符集编码。 #### 四、理解NLS_LANG参数 - **NLS_LANG**:该环境...
- 最后,使用`ALTER DATABASE CHARACTERSET`命令来修改字符集: ```sql ALTER DATABASE CHARACTERSET ZHS16GBK; ``` 2. **注意事项** - 修改字符集可能导致数据丢失,因此建议备份数据并在测试环境中尝试修改...
在一个节点上启动数据库实例,以便在不干扰其他节点的情况下进行字符集更改: ```bash [oracle@dbserver1 bin]$ ./srvctl start instance -d ltaiswk -i ltaiswk1 ``` 4. **验证字符集更改**: 连接到刚刚...
- 使用`ALTER DATABASE CHARACTERSET`命令进行修改,但不是所有的字符集都可以互相转换。 - 示例: ```sql ALTER DATABASE CHARACTER SET AL32UTF8; ``` 2. **修改客户端字符集**: - 修改客户端的NLS_LANG...
- 查询会话字符集:通过执行`SELECT value FROM nls_session_parameters WHERE parameter = 'NLS_CHARACTERSET'`,从v$nls_parameters获取当前会话的字符集环境。 4. 解决字符集问题的方法 - 在配置数据库时,确保...
- **CHARACTERSET**:控制客户端应用程序使用的字符集。通常情况下,这个字符集设置应与客户端(如Windows)的代码页相匹配,或对于Unicode应用设置为UTF8。 #### 二、查看 NLS_LANG 的方法 - **Windows**: - ...
- **NLS_CHARACTERSET**:指定客户端的字符集,不可在会话级别设置,只能通过修改`NLS_LANG`环境变量来设定。对于中文环境,通常设置为`ZHS16GBK`或`AL32UTF8`等支持中文的字符集。 查询`NLS_LANG`参数的方法是使用...
ALTER SYSTEM SET NLS_CHARACTERSET = '新的字符集' SCOPE=SPFILE; ``` 请注意,这只会改变初始化参数文件(SPFILE)的设置,实际字符集的更改可能需要重启数据库服务。 三、Oracle字符集的导入导出 在进行数据...
例如,`NLS_CHARACTERSET` 参数显示了数据库的默认字符集,这里是 `ZHS16GBK`,表明这是一个支持中文的字符集。`NLS_NCHAR_CHARACTERSET` 参数则表示 National Character Set,通常用于存储双字节字符,如 Unicode ...
当数据库的字符集设置(如`NLS_LANGUAGE`、`NLS_TERRITORY`和`NLS_CHARACTERSET`)与客户端应用程序或操作系统之间的编码设置不匹配时,就可能出现中文乱码的问题。例如,如果数据库设置为`ZHS16GBK`,而客户端工具...
服务端字符集的构成体现在数据字典表 V$NLS_PARAMETERS 的 NLS_LANGUAGE、NLS_TERRITORY、NLS_CHARACTERSET 三项取值上,其中 NLS_CHARACTERSET 的取值就是具体的数据库字符集。如利用查询语句 SQL>SELECT* FROM V$...
在OracleXE环境中,数据库导入过程...总的来说,OracleXE的导入和字符集更改涉及多个层面,包括数据库对象的迁移、字符集的适配以及PL/SQL绑定文件的处理。理解和掌握这些知识点对于维护和管理OracleXE数据库至关重要。
可以通过SQL查询`SELECT * FROM nls_database_parameters WHERE parameter = 'NLS_CHARACTERSET'`来查看服务器字符集,以及在客户端使用`SHOW PARAMETER NLS_CHARACTERSET`命令(在SQL*Plus中)或查询相应的环境变量...
- 同样地,`NLS_CHARACTERSET` 参数指定了当前会话的字符集。 - 这个命令对于理解当前用户的会话设置非常有帮助。 **4. 查看国家字符集:** ```sql SELECT * FROM nls_database_parameters WHERE parameter = ...
4. **禁用后台进程**:通过`ALTER SYSTEM SET JOB_QUEUE_PROCESSES=0`和`ALTER SYSTEM SET AQ_TM_PROCESSES=0`禁用作业队列和异步通知进程,以避免在字符集更改过程中出现问题。 5. **更改字符集**:尝试使用`ALTER ...
首先,你需要确定每个数据库实例的具体字符集,这可以通过查询数据库的`NLS_CHARACTERSET`参数来完成,例如: ```sql SELECT * FROM v$nls_parameters WHERE parameter = 'NLS_CHARACTERSET'; ``` 一旦确定了字符...
### 修改Oracle数据库字符集的方法 #### 背景与意义 在使用Oracle数据库的过程中,可能会遇到需要更改数据库字符集的情况。这通常发生在原有字符集不能满足新的业务需求时,例如需要支持更多的语言或特殊字符。...