oracle varchar2 字符存储长度问题

转载 2011年10月20日 11:24:44

 问题描述:

Oracle源表中可存放“中文测试”4个汉字,但经过PowerCenter抽取写入同结构目标表后,中文被截断为2个汉字“中文”。

CREATE TABLE 源表

(

  A  VARCHAR2(4 CHAR)

)

 

CREATE TABLE 目标表

(

  A  VARCHAR2(4 CHAR)

)

 

解决方案:

该问题是由NLS_LENGTH_SEMANTICS=BYTE所致,须将其设置为CHAR

 

如:为Integration Service添加 Environment variables:

Name: NLS_LENGTH_SEMANTICS

Value: CHAR

 

系统环境:
操作系统:Window Server 2003 SP2
中间件:WebSphere Platform 6.1
数据库:Oracle 10.2.0.1.0
字符集:NLS_CHARACTERSET UTF8
Maximo版本号:V7110-890

问题现象:
 1、在迁移Maximo数据库配置信息时,在执行类似语句时:
 insert into wfassignment@orcl57 select * from wfassignment t where t.processname=pc_row.processname and t.processrev=pc_row.processrev;
 总是报异常:description 实际长度是106,而目标表中的description字段长度为100,超出最大长度。

 2、结果在目标中检查字段:wfassignment.description varchar2(100) 只能插入约33个中文字加一个英文字;
 一个中文字占三个字节,一个英文字占一个字节。
 而原数据库wfassignment.description中可以插入100个汉字。

 3、用lengthb(description), length(description)查询字段长度,
  在原库中如果字段值都是汉字,则查询结果lengthb是length的三倍,
  在新库中同样的字段值,则查询结果长度一样。

问题原因分析:
 1、数据库字符集选择的是NLS_CHARACTERSET=UTF8,如果NLS_CHARACTERSET=ZHS16GBK就不会有这种情况;

 2、原库中NLS_LENGTH_SEMANTICS=CHAR,新库中NLS_LENGTH_SEMANTICS=BYTE;

 3、主要是第2条造成的,如果字符集是UTF8,字段类型VARCHAR2中一个字符占几个字节是由NLS_LENGTH_SEMANTICS参数决定的,
    如果是BYTE,则一个字符占一个字节,如果是CHAR,则一个字符占四个字节。
    通过下面的语句可以查看实际的data_length长度
    select table_name, column_name, data_type, data_length from cols

 4、查看数据库字符集参数的参数表是:v$nls_parameters,与字符集相关参数都在此表中;
  也可以用下面的语句查询;
  select * from v$parameter where name = 'nls_length_semantics'
  国家字符集:NLS_NCHAR_CHARACTERSET=AL16UTF16
  数据库字符集:NLS_CHARACTERSET=AL32UTF8

解决方案:
 一、修改目标数据库的"nls_length_semantics"属性值为"CHAR"
  
  1、执行:alter system set nls_length_semantics = CHAR scope=both;
   这种方式修改后重启数据库实例后,再查询时值又是BYTE了;
  2、修改文件:{installpath}/oracle/product/10.2.0/admin/orcl/pfile/init.ora.622010124846
   添加下面一段:
   ###########################################
   # NLS
   ###########################################
   nls_language="SIMPLIFIED CHINESE"
   nls_length_semantics=CHAR
   nls_territory="CHINA"

   这种方式修改后重启数据库实例后,再查询时值又是BYTE了;
  3、重启服务器电脑,然后查询时值是CHAR了,没有找到原因为什么重启电脑可以了。

 二、字段description还是不能插入100个汉字
  1、虽然nls_length_semantics已经改为CHAR了,当目标库中表字段description varchar2(100)还是
   不能插入100个汉字
  
  2、查询cols表,发现data_length长度还是100,再看wfassignment.description
   字段定义对象,发现数据库类型变为:varchar2(100 byte)
   
  3、alter table wfassignment modify description VARCHAR2(100); 问题解决;
   查询cols表,发现data_length长度是400了,可以插入100个汉字了。

  4、执行下面的语句,把生成的脚本保存为一个sql文件,用PL/SQL工具执行,就可以更新所有字符字段:
   select 'alter table '||table_name||' modify '||column_name||' VARCHAR2('||data_length||');'
   from cols where data_type = 'VARCHAR2' and table_name in
   (select table_name from tabs where status = 'VALID')

 三、如果是在Maximo产品中,因为字段wfassignment.description varchar2(100)都是全文检索字段,
  不能够直接修改字段长度,要求删除全文检索索引,处理方法见《删除Maximo中的所有全文检索索引

 

 

转载http://blog.csdn.net/lanyunit/article/details/5768581

 

 select * from v$nls_parameters

UPDATE PROPS$ SET NLS_CHARACTERSET='UTF8'
客户端的nls_lang也设为UTF8

 

http://www.eygle.com/index-special.htm oracle 数据恢复等

一个关于#include的问题

一个关于#include的问题 赵湘宁 问题:    我试图将一个头文件包含在工程的资源文件中。为此在.rc文件中添加了如下一行代码:#include "MyFile.h"虽然运行没有问题,但是每次在...
  • ghj1976
  • ghj1976
  • 2001-05-03 12:04:00
  • 1699

Oracle带输入输出参数的存储过程

原文链接:http://blog.itpub.net/29485627/viewspace-1248721/ (一)使用输入参数 需求:在emp_copy中添加一条记录,empno为已有empno...
  • hellochenlu
  • hellochenlu
  • 2016-08-13 17:28:17
  • 4489

Oracle数据库的varchar2(2)存储一个汉字时提示插入汉字过长问题

昨天在oracle实验课上出现了如题所示的问题:varchar2(2)在存储一个汉字时,提示插入汉字过长,要把它改成3才能存储1个汉字。于是百度google之。       先做以下实验:    ...
  • uestcong
  • uestcong
  • 2012-03-13 14:42:13
  • 13582

Oracle定义varchar2()类型存储汉字的长度问题

oracle 的varchar2(4000)通过jdbc的thin驱动连接为什么只可以存666个汉字? 谁说只能存储666个汉字的? varchar2最大是4000字节,那...
  • yaba213
  • yaba213
  • 2012-03-13 21:41:53
  • 19400

Oracle中,将VARchar2类型的字符串写入BLOB类型的字段中

1、在数据库中建一个新表用于测试。 create tbale tb_test (     id number,     blb blob ); commit; 2、往tb_test表中...
  • alvenz
  • alvenz
  • 2012-11-13 10:12:33
  • 470

获取oracle数据库表定义的字段长度并验证数据是否超出长度时需要了解的知识

最近项目上有个问题是需要获取oracle数据库中表定义的字段长度,然后进行验证将要输入到该数据库表的数据是否超出字段定义的长度。所以对oracle存储字符串的类型进行了解。 1、首先了解varchar...
  • a1s2d3q
  • a1s2d3q
  • 2015-09-21 18:28:13
  • 2143

Oracle存储过程及参数理解

一、过程 (存储过程)     过程是一个能执行某个特定操作的子程序。使用CREATE OR REPLACE创建或者替换保存在数据库中的一个子程序。 示例1:声明存储过程,该过程返回dept表行数 ...
  • chenleixing
  • chenleixing
  • 2015-01-07 14:38:45
  • 2949

Oracle存储过程及举例(几种参数情况的存储过程)

Oracle存储过程及举例(几种参数情况的存储过程)
  • u013882957
  • u013882957
  • 2017-05-06 22:22:50
  • 5753

带输入参数的Oracle的存储过程

no_data_found是Oracle自己预先定义的一个exception过程:用户执行特定的操作,当建立过程时既可以指定输入参数(in),也可以指定输出参数(out)在过程中使用输入参数,可以将数...
  • Dean_Deng
  • Dean_Deng
  • 2011-07-17 19:45:34
  • 3070

Oracle存储过程【带输入输出参数】

 create or replace procedure proc_emp(eno in number,sal out number)--带输入输出参数的存储过程asemp_records emp%r...
  • ht_927
  • ht_927
  • 2010-11-24 12:20:00
  • 15701
收藏助手
不良信息举报
您举报文章:oracle varchar2 字符存储长度问题
举报原因:
原因补充:

(最多只允许输入30个字)