NHibernate映射oracle數據庫CLOB、BLOB類型大數據字段

最初始最簡單的方法以下:數據庫

首先建一個測試類:c#

public class testClass
  {
     //須要映射爲CLOB類型的字段
     public string WKT{get;set;}
  }

nhibernate配置文件中:sass

<property column="WKT" type="StringClob" name="WKT" length="200000" />

具體nhibernate的配製方法什麼的不在此贅述,請查看本人nhibernate相關博客。session

番外:CLOB、BLOB字段最大上限爲4GB,即4x1024x1024x1024個字節,除非特別須要,不建議length過大,合適就行。我取的20萬字節oracle

寫一個簡單的方法插入數據測試:
ide

var testClass = new testClass();
testClass.WKT = data; // data是一個變量,存儲你要插入的字符串
ISession session = SessionFactory.OpenSession();
session.Save(testClass);
session.Flush();

咱們會發現此時oracle數據庫WKT字段爲NCLOB類型,可是測試數據在2000字節如下以及4000字節以上時順利經過,在2000-4000字節時發生錯誤【ORA-01461:僅能夠插入LONG列的LONG值賦值】,具體緣由聽大神說貌似是oracle.client的問題,在此不作詳細解釋。測試

具體解決方案以下:
spa

相信你們作nhibernate映射時都會遇到自定義類型的狀況,如不會請參考本人相關nhibernate微博。那麼咱們也能夠爲CLOB自定義一個類型來設置正確的oracleType:hibernate

首先添加一個針對LOB字段的補丁類(若是隻涉及到CLOB字段,也能夠直接添加針對CLOB的類,一步到位):code

PatchForOracleLobField.cs

using System;
using System.Collections.Generic;
using System.Data;
using System.Text;
using NHibernate;
using NHibernate.SqlTypes;
using NHibernate.UserTypes;
namespace Test.type
{
    public abstract class PatchForOracleLobField : IUserType
    {
        public bool IsMutable
        {
            get { return false; }
        }
        public System.Type ReturnedType
        {
            get { return typeof(StringClobSqlType); } //注意此處
        }
        public SqlType[] SqlTypes
        {
            get
            {
                return new SqlType[] { NHibernateUtil.StringClob.SqlType };  //注意此處
            }
        }
        public object DeepCopy(object value)
        {
            return value;
        }
        public new bool Equals(object x, object y)
        {
            return x == y;
        }
        public int GetHashCode(object x)
        {
            return x.GetHashCode();
        }
        public object Assemble(object cached, object owner)
        {
            return DeepCopy(cached);
        }
        public object Disassemble(object value)
        {
            return DeepCopy(value);
        }
        public object NullSafeGet(IDataReader rs, string[] names, object owner)
        {
            return NHibernate.NHibernateUtil.StringClob.NullSafeGet(rs, names[0]);
        }
        public abstract void NullSafeSet(IDbCommand cmd, object value, int index);
        public object Replace(object original, object target, object owner)
        {
            return original;
        }
    }
}

再添加一個針對CLOB的類:

OracleClobField.cs

using System;
using System.Collections.Generic;
using System.Data;
using System.Data.OracleClient;
using System.Text;
namespace Test.type
{
    public class OracleClobField : PatchForOracleLobField
    {
        public override void NullSafeSet(IDbCommand cmd, object value, int index)
        {
            if (cmd is OracleCommand)
            {
                OracleParameter param = cmd.Parameters[index] as OracleParameter;
                if (param != null)
                {
                    param.OracleType = OracleType.Clob;  //注意此處
                    param.IsNullable = true;
                }
            }
            NHibernate.NHibernateUtil.StringClob.NullSafeSet(cmd, value, index);
        }
    }
}

而後咱們修改nhibernate的映射文件:

修改前:

<property column="WKT" type="StringClob" name="WKT" length="200000" />

修改後:

<property column="WKT" type="命名空間+testClass, testClass所在程序集" name="WKT" length="200000" />

再次調用測試方法進行測試,發現問題解決!!!

注:相信你們都知道nhibernate映射時,表已存在則只添加不修改,因此建議刪庫後重建表測試!

相關文章
相關標籤/搜索