使用NHibernate中的Parent -Child关系处理集合

时间:2010-07-14 14:59:54

标签: nhibernate collections

当我处理两个列表(包括旧的chidren(来自db))和另一个带有修改后的子列表(来自用户输入)时,我遇到了一个问题,即找出要添加,更新和删除的子项。

目前我必须遍历列表并进行比较以找出差异。

有没有人知道更好的方法,而不必遍历列表?

1 个答案:

答案 0 :(得分:6)

IEnumerable.Except扩展方法非常适用于此。在大多数情况下,除非您在对象中仔细重写了Equals和GetHashCode,否则您将希望使用接受IEqualityComparer<T>的重载来进行身份比较。

鉴于

IEnumerable<Company> oldList, IEnumerable<Company> newList

代码是:

var companiesToAdd = newList.Except(oldList).ToArray();
var companiesToRemove = oldList.Except(newList).ToArray();

ToArray()调用是存在的,这样您就可以在迭代添加和删除列表时对原始列表进行更改。

编辑:以下是两个实用程序类,使此操作变得简单。用法是

var diff = new DiffIEnumerable<Company>(oldList, newList, x => x.CompanyId);
var companiesToAdd = diff.InYNotX;
var companiesToRemove = diff.InXNotY;

public class IdentityComparer<T> : IEqualityComparer<T> where T : class
{
    private readonly Func<T, object> _getIdentity;

    public IdentityComparer(Func<T, object> getIdentity)
    {
        _getIdentity = getIdentity;
    }

    public bool Equals(T x, T y)
    {
        if (x == null || y == null)
        {
            return false;
        }
        return _getIdentity(x).Equals(_getIdentity(y));
    }

    public int GetHashCode(T obj)
    {
        return _getIdentity(obj).GetHashCode();
    }
}

public class DiffIEnumerable<T> where T : class
{
    public DiffIEnumerable(IEnumerable<T> x, IEnumerable<T> y, Func<T, object> getIdentity) :
        this(x, y, new IdentityComparer<T>(getIdentity))
    { }

    public DiffIEnumerable(IEnumerable<T> x, IEnumerable<T> y, IEqualityComparer<T> comparer)
    {
        InXAndY = x.Intersect(y, comparer).ToArray();
        InXNotY = x.Except(y, comparer).ToArray();
        InYNotX = y.Except(x, comparer).ToArray();
    }

    public IEnumerable<T> InXAndY { get; private set; }

    public IEnumerable<T> InXNotY { get; private set; }

    public IEnumerable<T> InYNotX { get; private set; }

}