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中的部分内容,就可以了。

 

 

主要研究技术:代码生成工具、Visio二次开发、送水管理软件等共享软件开发
专注于Winform开发框架、WCF开发框架的研究及应用。
  转载请注明出处:
撰写人:伍华聪  http:
//www.iqidi.com 
    

posted on 2010-02-05 12:15 伍华聪 阅读(2361) 评论(3) 编辑 收藏

评论

#1楼  回复 引用 查看   

这样搞你不累吗,你把你的字段用一个类表示,再转换成xml,存到这个clob就可以了,看你这累死人了
2010-02-05 12:56 | zhdw      

#2楼  回复 引用 查看   

这个广告绕的太大了一些。
2010-02-05 13:37 | 古道轻风      

#3楼  回复 引用 查看   

学习了。
2010-02-05 15:32 | 潇湘隐者      

导航

<2010年2月>
31123456
78910111213
14151617181920
21222324252627
28123456
78910111213

公告

昵称:伍华聪
园龄:6年7个月
荣誉:推荐博客
粉丝:656
关注:22

搜索

 

常用链接

最新随笔

我的标签

随笔档案(199)

文章分类(14)

文章档案(15)

相册

我的好友

我的技术支持站点(http://www.iqidi.com)

积分与排名

  • 积分 - 624413
  • 排名 - 79

最新评论

阅读排行榜

评论排行榜

推荐排行榜