nhibernate – 如何将对象从Reflection转换为泛型集合?

我正在尝试编写一个Compare方法来比较某些POCO中使用Reflection的属性,以确保它们已经正确地保存到数据库中.例如,假设我有这个POCO:

public class NoahsArk
{
    public string Owner { get; set; }
    public ICollection<Animal> Animals { get; set; }
}

我想要做的是:

[Test]
public class Saves_Correctly_To_Database()
{
    var noahsArk = new NoahsArk { // some setup code here };
    db.Save(noahsArk);
    var dbNoahsArk = db.Get<NoahsArk>(noahsArk.Id);

    Assert.That(Compare(noahsArk, dbNoahsArk), Is.True);
}

我正在使用的ORM是NHibernate.到目前为止,我的Compare方法看起来像这样:

public static bool EqualsProperties<T>(this T x, T y)
{
    var xType = x.GetType();

    foreach (var property in xType.GetProperties())
    {
        if (property.GetValue(x, null).Implements(typeof(ICollection<>)))
        {
            var xValue = property.GetValue(x, null) as ICollection<T>;
            var yValue = property.GetValue(y, null) as ICollection<T>;
        }

Object.Implements()是我编写的一个扩展方法,用于确定类型是否实现了接口.如您所见,该方法不完整.我遇到的问题是当我使用property.GetValue(x,null)时,它返回一个对象,我不知道如何将它转换为特定的通用ICollection类型.我需要能够这样做,所以我可以使用LINQ来执行x.Contains(y)来比较两个集合的相等性.有关如何做到这一点的任何想法?

附:我尝试使用Compare .NET Objects,但它在NHibernate内部的某个地方给了我一个空引用异常.它没有正确处理NHibernate如何代表延迟加载的ICollection.更糟糕的是,NHibernate修改POCO以支持延迟加载,但这一切都是在运行时完成的.在源代码中,看起来您只是使用常规ICollection,但NHibernate在运行时将其更改为NHibernate.Collections.Generic.PersistentSet,这就是导致比较器失败的原因.

最佳答案 您的问题有点令人困惑,因为您在EqualsProperties方法的声明中不需要类型参数T.您只需要

public static bool EqualsProperties(this object x, object y)

然后,您继续使用相同的参数T将x和y的属性转换为ICollection< T&gt ;;但是,这些集合中的对象显然可能具有与x和y不同的类型. 现在回答你的问题:你不需要转换为正确的泛型类型来使用LINQ Contains方法.你可以这样做:

xValue = property.GetValue(x, null);
yValue = property.GetValue(y, null);
if (typeof(IEnumerable).IsInstanceOf(x))
{
   IEnumerable<object> xEnumerable = (x as IEnumerable).Cast<object>();
   IEnumerable<object> yEnumerable = (y as IEnumerable).Cast<object>();
   // use any LINQ method you like now
}

您还应该确保使用采用相等比较器的LINQ重载,因为您的域对象显然不会覆盖Equals方法本身.否则你不会编写这个单元测试代码来比较它们.

点赞