zoukankan      html  css  js  c++  java
  • 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开发框架/混合式开发框架Web开发框架Bootstrap开发框架微信门户开发框架的研究及应用
      转载请注明出处:
    撰写人:伍华聪  http://www.iqidi.com 
        
  • 相关阅读:
    Java中String.valueOf()方法的解释
    React 16.13.1开启严格模式会触发两次render
    React开发中使用react-router-dom路由最新版本V5.1.2(三)路由嵌套子路由
    React开发中使用react-router-dom路由最新版本V5.1.2(二)路由跳转及传参
    React开发中使用react-router-dom路由最新版本V5.1.2(一)基本跳转属性
    vue 开发中实现provide和inject的响应式数据监听
    VUE开发中,数据变化后,修改了绑定对象的属性值后,页面dom没有发生变化,两种方法可以重新渲染$set和$forceUpdate
    VUE开发中.sync 修饰符的作用
    VUE开发中.native 修饰符的作用
    git报错:CONFLICT (modify/delete) ,git冲突,解决项目开发中改了相同文件同一行
  • 原文地址:https://www.cnblogs.com/wuhuacong/p/1664140.html
Copyright © 2011-2022 走看看