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

本文详细解析了Oracle数据库中VARCHAR2类型的特性与限制,包括不同字符集下VARCHAR2字段的存储方式,以及通过JDBC Thin驱动连接时可能出现的问题,并提供了解决方案。

首先查找了一下varchar2()的一些基本资料 

1.varchar2把所有字符都占两字节处理(一般情况下),varchar只对汉字和全角等字符占两字节,数字,英文字符等都是一个字节;
2.VARCHAR2把空串等同于null处理,而varchar仍按照空串处理;
3.VARCHAR2字符要用几个字节存储,要看数据库使用的字符集,
大部分情况下建议使用varchar2类型,可以保证更好的兼容性。

oracle 的varchar2(4000)通过jdbc的thin驱动连接为什么只可以存666个汉字?

谁说只能存储666个汉字的?

varchar2最大是4000字节,那么就看你的oracle字符集,如果字符集是16位编码的,ZHS16GBK,那么每个字符16位,2字节,所以可以容纳2000字符。
如果是32位编码的字符集,那么只能存储 1000个字符。

从后面的例子的 length(字符长度)和lengthb(字节长度)的差别就可以看出来了。
create table T_BOARD( NAME      VARCHAR2(50)

insert into      T_BOARD      values ('测试');

select      t.name,length(t.name),lengthb(t.name) from t_board t

20090322131055_1.jpg

字符集合决定varchar2的长度

问题描述:
                            intert into T_BOARD      values('超过17个汉字');
                           报错:插入字符过长!发现一个汉字占3个字节,所以报错!!!

问题所在:
                          使用的字符集是UTF8,就有可能出现这个错误!
                          使用命令查看:

SQL> select parameter,value from nls_database_parameters where parameter like 'NLS_CHARACTERSET';

PARAMETER
------------------------------------------------------------
VALUE
--------------------------------------------------------------------------------
NLS_CHARACTERSET
AL32UTF8

解决方法:
                          建议使用ZHS16GBK字符集!
                          操作:  
                                  SQL> SHUTDOWN IMMEDIATE;  
                                  SQL> STARTUP MOUNT; 
                                  SQL> ALTER SYSTEM ENABLE RESTRICTED SESSION; 
                                  SQL> ALTER SYSTEM SET JOB_QUEUE_PROCESSES=0; 
                                  SQL> ALTER DATABASE OPEN; 
                                  SQL> ALTER DATABASE CHARACTER SET AL32UTF8/ZHS16GBK;

                                  SQL> SHUTDOWN IMMEDIATE;


                                  SQL> STARTUP;
                
问题解决!!! (改变时注意新的字符集是旧的超集问题)

附注1:

Re: 研究:为什么我的ORACLE的数据操作突然报数量限制的错误(怪)?发表时间: 2003-12-05 09:25
回复此消息
这是很经典的OracleJDBC问题。一般直接用ps.setString()设置字符串数据时,Oracle的JDBC驱动会将中文转换为2字节或3字节,不固定的,因此经常会越界。
如果你改为ps.setCharacterStream()就是固定的每个中文两个字节 

根据oracle的文档,thin的jdbc驱动,会根据字符集合决定varchar2的长度,如果不是ascii 或者拉丁字符集合,长度的限制就是2000,因为它认为其他字符集都需要两个字节来存储,但是通过jdbc的setString方法时候,驱动会把 java的utf-16转换为utf-8,这样英文由两个字节变成一个字节,中文由两个字节变为3个字节,所以2000/3大概就是666个中文字符了。

附注2:

在做更新操作时,办法:
pstmt.setBytes(4,aimString.getBytes());
取数据时:
new String(resultSet.getBytes("fieldName"),"ISO-8859-1");

转载于:https://my.oschina.net/i33/blog/74951

<think> 我们正在讨论Oraclevarchar2数据类型的最大长度问题。根据提供的引用,我们可以总结如下: 1. 在表字段定义中,varchar2的最大长度允许为4000字节(无论是varchar2(4000 byte)还是varchar2(4000 char))[^1][^3][^4]。 2. 但是,当使用char作为单位时(如varchar2(4000 char)),表示最多可以存储4000个字符,但实际存储的字节数不能超过4000字节(在数据库块大小为标准8k时)[^4]。 3. 在PL/SQL中,变量声明为varchar2类型时,最大长度可以达到32767字节[^3]。 4. 对于函数参数,varchar2类型也可以达到32767字节;但函数返回值类型varchar2的最大长度仍然是4000字节[^3]。 另外,关于存储汉字: - 在默认情况下(使用字节单位),每个汉字占用2个或3个字节(取决于数据库字符集,如ZHS16GBK为2字节,UTF-8为3字节)[^4]。 - 因此,如果使用varchar2(10 byte),最多可以存储5个汉字(在ZHS16GBK字符集下)或3个汉字(在UTF-8字符集下,因为3*3=9<10,3个汉字占9字节,而4个汉字则12字节>10)。 - 如果使用varchar2(10 char),则可以存储10个汉字,但总字节数不能超过4000。所以,即使指定为4000 char,实际插入的汉字也不能超过1333个(在UTF-8字符集下,因为4000/3≈1333.33)[^4]。 因此,总结Oraclevarchar2的最大长度: - 作为表列:最大4000字节。 - 作为PL/SQL变量:最大32767字节。 注意:Oracle 12c及以上版本引入了扩展数据类型(通过MAX_STRING_SIZE参数控制),允许将varchar2的最大长度扩展到32767字节。但需要特别配置,且一旦扩展不可逆。 现在,根据用户需求,他想要查询Oraclevarchar2数据类型的最大长度,我们可以这样回答: </think> ### Oracle VARCHAR2 数据类型最大长度详解 在 Oracle 数据库中,`VARCHAR2` 的最大长度限制取决于使用场景和定义方式,以下是详细总结: #### 1. **表字段定义(列存储)** - **最大字节数**:4000 字节(硬性限制) - **定义方式**: - `VARCHAR2(n BYTE)`:按字节计算,支持最多 4000 字节 - `VARCHAR2(n CHAR)`:按字符计算,但实际存储字节数仍不能超过 4000 字节[^3][^4] - **汉字存储示例**: - UTF-8 字符集:单个汉字占 3 字节 - `VARCHAR2(1333 CHAR)`:可存 1333 个汉字(1333×3≈4000 字节) - 尝试存储 1334 个汉字(4012 字节)会报错 `ORA-01461`[^4] #### 2. **PL/SQL 编程环境** | 场景 | 最大长度 | 说明 | |---------------------|----------|--------------------------| | 变量声明 | 32,767 字节 | 如 `DECLARE str VARCHAR2(32767)`[^3] | | 函数参数 | 32,767 字节 | 参数传递不受 4000 字节限制[^3] | | 函数返回值 | 4000 字节 | 返回给 SQL 时仍受表字段限制[^3] | #### 3. **特殊场景与注意事项 - **扩展数据类型Oracle 12c+)**: 通过修改参数可实现更大存储: ```sql ALTER SYSTEM SET MAX_STRING_SIZE = EXTENDED; ``` 启用后最大支持 32,767 字节(需重启数据库)[^1] - **字节 vs 字符差异**: ```sql -- 按字节定义(默认) CREATE TABLE t1 (col VARCHAR2(10 BYTE)); -- 最多存 10 字节数据 -- 按字符定义 CREATE TABLE t2 (col VARCHAR2(10 CHAR)); -- 最多存 10 个字符 ``` 存储汉字时: - `t1` 在 UTF-8 下最多存 3 个汉字(3×3=9 字节) - `t2` 可存 10 个汉字(但总字节不能超 4000)[^2][^4] #### 4. **常见错误解决方案 | 错误代码 | 原因 | 解决方案 | |---------------|------------------------------|----------------------------| | ORA-00910 | 定义长度 >4000 字节 | 减小字段长度或启用扩展模式 | | ORA-12899 | 插入数据超过定义长度 | 检查字符集和单位(BYTE/CHAR)| | ORA-01461 | 插入数据超过 4000 字节 | 拆分数据或使用 CLOB 类型 | > **最佳实践**:对超过 4000 字节的数据使用 `CLOB` 类型,对多语言系统优先使用 `CHAR` 单位定义
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值