Oracle中对超长内容的处理,避免出现can bind a LONG value only for insert into a LONG column错误

本人在做一个新闻内容的模块的时候,发现如果内容在4K以上的字符串会出错,得到的内容会是乱码(也就是被自动截断),如果小于4K,那么就不会有问题。

原来采用了Varchar2的类型来存储,但发现后修改为Clob类型的也出现同样的问题,而且发现日志的错误是:System.Data.OracleClient.OracleException: ORA-01461: can bind a LONG value only for insert into a LONG column。

由于Clob类型是可以存放很大类型的文本数据的,不会是数据库字段容量不够,因此检查插入和更新的C#代码,发现原来的代码是这样的:

        public bool Insert(Hashtable recordField, string targetTable, DbTransaction trans)
        {
            bool result = false;
            string fields = ""; // 字段名
            string vals = ""; // 字段值
            if ( recordField == null || recordField.Count < 1 )
            {
                return result;
            }

            List<OracleParameter> paramList = new List<OracleParameter>();
            IEnumerator eKeys = recordField.Keys.GetEnumerator();

            while ( eKeys.MoveNext() )
            {
                string field = eKeys.Current.ToString();
                fields += field + ",";
                if (!string.IsNullOrEmpty(seqField) && !string.IsNullOrEmpty(seqName)
                    && (field.ToUpper() == seqField.ToUpper()))
                {
                    vals += string.Format("{0}.NextVal,", seqName);
                }
                else
                {
                    vals += string.Format(":{0},", field);
                    object val = recordField[eKeys.Current.ToString()];
                    paramList.Add(new OracleParameter(":" + field, val));
                }
            }

            fields = fields.Trim(',');//除去前后的逗号
            vals = vals.Trim(',');//除去前后的逗号
            string sql = string.Format("INSERT INTO {0} ({1}) VALUES ({2})", targetTable, fields, vals);

            Database db = DatabaseFactory.CreateDatabase();
            DbCommand command = db.GetSqlStringCommand(sql);
            command.Parameters.AddRange(paramList.ToArray());

            if ( trans != null )
            {
                result = db.ExecuteNonQuery(command, trans) > 0;
            }
            else
            {
                result = db.ExecuteNonQuery(command) > 0;
            }

            return result;
        }

 重要的地方就是我使用了该行代码:

paramList.Add(new OracleParameter(":" + field, val));

猜想可能是由于这行代码的问题导致,因此修改参数化的Oracle参数变量代码为另外一个种指定对象类型的方式:

OracleParameter a = new OracleParameter(":" + field, OracleType.Clob, val.ToString().Length);
                        a.Value = val;
                        paramList.Add(a);

验证通过,发现再长的内容,写入也是正常的,不会出错和出现的截断乱码。由于第一种忽略了Oracle参数类型,就是为了适应各种类型对象的参数化构造,实现统一添加参数化内容的,由于超长的字符内容会出现问题,因此只好修改基类操作的添加参数代码,添加一个条件分支作为处理。调整后的插入代码如下(更新代码类似操作):

        public bool Insert(Hashtable recordField, string targetTable, DbTransaction trans)
        {
            bool result = false;
            string fields = ""; // 字段名
            string vals = ""; // 字段值
            if ( recordField == null || recordField.Count < 1 )
            {
                return result;
            }

            List<OracleParameter> paramList = new List<OracleParameter>();
            IEnumerator eKeys = recordField.Keys.GetEnumerator();

            while ( eKeys.MoveNext() )
            {
                string field = eKeys.Current.ToString();
                fields += field + ",";
                if (!string.IsNullOrEmpty(seqField) && !string.IsNullOrEmpty(seqName)
                    && (field.ToUpper() == seqField.ToUpper()))
                {
                    vals += string.Format("{0}.NextVal,", seqName);
                }
                else
                {
                    vals += string.Format(":{0},", field);
                    object val = recordField[eKeys.Current.ToString()];

                    if (val.ToString().Length >= 4000)
                    {
                        OracleParameter a = new OracleParameter(":" + field, OracleType.Clob, val.ToString().Length);
                        a.Value = val;
                        paramList.Add(a);
                    }
                    else
                    {
                        paramList.Add(new OracleParameter(":" + field, val));
                    }
                }
            }

            fields = fields.Trim(',');//除去前后的逗号
            vals = vals.Trim(',');//除去前后的逗号
            string sql = string.Format("INSERT INTO {0} ({1}) VALUES ({2})", targetTable, fields, vals);

            Database db = DatabaseFactory.CreateDatabase();
            DbCommand command = db.GetSqlStringCommand(sql);
            command.Parameters.AddRange(paramList.ToArray());

            if ( trans != null )
            {
                result = db.ExecuteNonQuery(command, trans) > 0;
            }
            else
            {
                result = db.ExecuteNonQuery(command) > 0;
            }

            return result;
        }

 这样,就可以在使用代码生成工具Database2Sharp(http://www.iqidi.com/database2sharp.htm)生成的Oracle代码中,不需要改变任何地方,只需要调整BaseDAL的基类中Insert和Update中的部分内容,就可以了。

本文转自博客园伍华聪的博客,原文链接:Oracle中对超长内容的处理,避免出现can bind a LONG value only for insert into a LONG column错误,如需转载请自行联系原博主。

时间: 2024-09-11 13:24:27

Oracle中对超长内容的处理,避免出现can bind a LONG value only for insert into a LONG column错误的相关文章

proc中插入VARCHAR2字段报ORA-01461: can bind a LONG value only for insert into a LONG column

最近做一个天信达货运接口的项目,碰到这么个诡异的问题. 背景: 使用proc写的应用,向数据库插入记录.表中有三个VARCHAR2(4000)类型的字段.注:Oracle 9i的库. 问题: 执行的时候提示:ORA-01461: can bind a LONG value only for insert into a LONG column 无法插入记录,但使用PLSQL Developer或SQLPLUS手工执行相同的SQL却没有问题. 然后换了一个10g的库,用proc可以正确完成插入. 难

Java使用JDBC或MyBatis框架向Oracle中插入XMLType数据_java

先来了解一下什么是XMLType类型. XMLType是Oracle从9i开始特有的数据类型,是一个继承了Blob的强大存在,可以用来存储xml并提供了相当多的操作函数.理论上可以保存2G大小的数据. 那怎么样通过java来插入XMLType类型的数据呢?项目当中采用的是Mybatis,总是出现莫名的异常,都搞不清楚到底是Mybatis的问题还是jdbc本身的问题,所以打算一步步来,先搞定jdbc,再解决Mybatis. JDBC在折腾了半天之后,发现jdbc操作主要有3种方法: 一.在Java

oracle中一些sql以及存储过程小积累(转)

将当前日期转换为上一个月   SELECT TO_CHAR(ADD_MONTHS(SYSDATE, -1), 'yyyymm') --获取当前时间的前一个月             FROM DUAL; select sysdate from dual; /**获取当前时间到秒**/ select  sysdate-3 from dual;/**获取当前2天**/ select round(sysdate)  as 格式成日期 from dual; select to_date('2008-9-

oracle中 查询时间管理

1.关于时间显示参数 SQL> alter session set nls_timestamp_format = 'yyyy-mm-dd hh24:mi:ss'; 会话已更改. SQL> select systimestamp from dual; SYSTIMESTAMP --------------------------------------------------------------------------- 14-3月 -07 05.00.33.599000 下午 +08:00

Oracle中的临时表用法汇总

oracle|临时表 说明:下文中的一些说明和示例代码摘自CSDN,恕不一一指明出处,在此一并对相关作者表示感谢! 如果作者有异议,请来信说明  1 语法 在Oracle中,可以创建以下两种临时表: 1) 会话特有的临时表 CREATE GLOBAL TEMPORARY <TABLE_NAME> (<column specification> ) ON COMMIT PRESERVE ROWS: 2) 事务特有的临时表 CREATE GLOBAL TEMPORARY <TAB

游标处理-oracle中怎么将一个包含多内容的字段修改其部分内容

问题描述 oracle中怎么将一个包含多内容的字段修改其部分内容 ![ s3_fbfw字段中包含yzcm字段的内容,现在要s3_fbfw修改中的yzcm部分改成xzcm,s3是main表,yzcm是temp表,求大神指导 解决方案 字符串截取就可以吧,substring这个函数吧好像

在oracle中查询xml表列内容显示不全,为什么???

问题描述 在oracle中查询xml表列内容显示不全,为什么??? 建立了一个包含xml列的表,也成功插入了xml项,但是用查询语句显示不全xml的内容,不知道为啥?? CREATE TABLE employees( eid int primary key, einfo xmltype ); insert into employees values(1,XMLType(bfilename('XMLDIR', 'einfoTest.xml'),nls_charset_id('GB2312') ))

ORACLE中LOB字段的使用和维护

oracle 摘要:本文通过实例介绍了在ORACLE数据库中通过DBMS_LOB包使用和维护LOB数据类型的基本方法. 关键词:ORACLE DBMS_LOB LOB 维护 中图分类号:TP31 1.引言 随着社会的发展,在现代信息系统的开发中,需要存储的已不仅仅是简单的文字信息,同时还包括一些图片和音像资料或者是超长的文本.比如开发一套旅游信息系统,每一个景点都有丰富的图片.音像资料和大量的文字介绍.这就要求后台数据库要有存储这些数据的能力.ORACLE公司在其Oracle8i中通过提供LOB

Oracle中的并行

Oracle中的并行 一Oracle中的并行 首先Oracle会创建一个进程用于协调并行服务进程之间的信息传递这个协调进程将需要操作的数据集例如表的数据块分割成很多部分称为并行处理单元然后并行协调进程给每个并行进程分配一个数据单元.例如有四个并行服务进程它们就会同时处理各自分配的单元当一个并行服务进程处理完毕后协调进程就会给它们分配另外的单元如此反复直到表上的数据都处理完毕最后协调进程负责将每个小的集合合并为一个大集合作为最终的执行结果返回给用户.并行处理的机制实际上就是把一个要扫描的数据集分成