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错误,如需转载请自行联系原博主。



目录
相关文章
|
6月前
|
Oracle 关系型数据库 数据库
Oracle中merge Into的用法
Oracle中merge Into的用法
|
SQL 存储 Oracle
oracle错误代码大全(超详细)
本篇文章是对oracle错误代码进行了详细的总结与分析,需要的朋友参考下
1667 0
|
6月前
|
Oracle 关系型数据库 数据库
Flink Sink to Oracle 存在字段CLOB类型,如何处理错误”ORA-01461: 仅能绑定要插入LONG的LONG值“
做Flink CDC同步数据过程中,目标是Oracle数据库,其中某个字段较大被设置为CLOB类型,其中会遇到异常,”ORA-01461: 仅能绑定要插入LONG的LONG值“
|
SQL Oracle 关系型数据库
解决Oracle的状态: 失败 -测试失败: IO 错误: The Network Adapter could not establish the connection
解决Oracle的状态: 失败 -测试失败: IO 错误: The Network Adapter could not establish the connection
2085 0
解决Oracle的状态: 失败 -测试失败: IO 错误: The Network Adapter could not establish the connection
|
Oracle 关系型数据库
Oracle中的wm_concat(column)函数
Oracle中的wm_concat(column)函数
|
存储 关系型数据库 MySQL
MySQL出现Data too long for column...(错误号1406)和 Data truncated for column...(错误号1265)
MySQL出现Data too long for column...(错误号1406)和 Data truncated for column...(错误号1265)
1000 0
MySQL出现Data too long for column...(错误号1406)和 Data truncated for column...(错误号1265)
Zp
|
SQL Oracle 关系型数据库
Oracle sql 批量插入 Insert all into
Oracle sql 批量插入 Insert all into
Zp
446 0
|
SQL Oracle 关系型数据库
【数据库】解决 oracle: SQL 错误 [900] [42000]: ORA-00900: 无效 SQL 语句
【数据库】解决 oracle: SQL 错误 [900] [42000]: ORA-00900: 无效 SQL 语句
2342 0
【数据库】解决 oracle: SQL 错误 [900] [42000]: ORA-00900: 无效 SQL 语句
|
存储 Oracle 关系型数据库
常见Oracle错误都在这了: ORA-00257/ORA-00313/ORA-28000/ORA-28000
常见Oracle错误都在这了: ORA-00257/ORA-00313/ORA-28000/ORA-28000
常见Oracle错误都在这了: ORA-00257/ORA-00313/ORA-28000/ORA-28000
|
SQL Oracle 关系型数据库
oracle Sql语句Date转long时间戳
oracle Sql语句Date转long时间戳
1289 0
oracle Sql语句Date转long时间戳