我试图通过代码使用NHibernate(v3.3)映射来获取自定义类型。我尝试了这个例子here,但没有运气。我想要实现的自定义类型是一个修剪来自数据库的字符串的类型。
我收到以下异常:
PropertyAccessException:无效的演员表(检查你的 属性类型不匹配的映射)。 {“无法将'System.String'类型的对象强制转换为'ConsoleApplication1.TrimmedString'。”}
以下是我的完整尝试(gist)。
public class TrimmedString : IUserType
{
public object NullSafeGet(IDataReader rs, string[] names, object owner)
{
//treat for the posibility of null values
string resultString = (string) NHibernateUtil.String.NullSafeGet(rs, names[0]);
if (resultString != null)
return resultString.Trim();
return null;
}
public void NullSafeSet(IDbCommand cmd, object value, int index)
{
if (value == null)
{
NHibernateUtil.String.NullSafeSet(cmd, null, index);
return;
}
value = ((string) value).Trim();
NHibernateUtil.String.NullSafeSet(cmd, value, index);
}
public object DeepCopy(object value)
{
if (value == null) return null;
return string.Copy((String) value);
}
public object Replace(object original, object target, object owner)
{
return original;
}
public object Assemble(object cached, object owner)
{
return DeepCopy(cached);
}
public object Disassemble(object value)
{
return DeepCopy(value);
}
public SqlType[] SqlTypes
{
get
{
SqlType[] types = new SqlType[1];
types[0] = new SqlType(DbType.String);
return types;
}
}
public Type ReturnedType
{
get { return typeof (String); }
}
public bool IsMutable
{
get { return false; }
}
public new bool Equals(object x, object y)
{
if (ReferenceEquals(x, y)) return true;
var xString = x as string;
var yString = y as string;
if (xString == null || yString == null) return false;
return xString.Equals(yString);
}
public int GetHashCode(object x)
{
return x.GetHashCode();
}
}
这是我的映射:
public class Person
{
public virtual int Id { get; set; }
public virtual TrimmedString FirstName { get; set; }
public virtual string LastName { get; set; }
}
public class PersonMap : ClassMapping<Person>
{
public PersonMap()
{
Table("Source");
Id(i => i.Id);
Property(i => i.FirstName, map => map.Type<TrimmedString>());
Property(i => i.LastName);
}
}
不确定我是否必须在NHibernate配置对象中做任何特殊操作,但我已将其包含在上面链接的Gist中。
答案 0 :(得分:5)
在Person
中,它应该是......
public virtual string FirstName { get; set; }
...,而不是TrimmedString
。 TrimmedString
只是指导NHibernate如何使该属性保持水合和脱水的类。它应用的属性应该是ReturnedType
指定的类型 - 换句话说,String
。 NHibernate试图将FirstName
属性设置为string
值(因为这是TrimmedString
所说的应该做的事情),但它不能,因为FirstName
只允许{ {1}} s,因此“无效投射”错误。