循环遍历一个对象并找到not null属性

时间:2012-11-03 19:05:43

标签: c# .net reflection

我有2个相同对象的实例,o1和o2。如果我正在做像

这样的事情
 if (o1.property1 != null) o1.property1 = o2.property1 

表示对象中的所有属性。循环遍历Object中所有属性的最有效方法是什么?我看到有人使用PropertyInfo检查nulll属性,但看起来他们只能通过PropertyInfo集合但不链接属性的操作。

感谢。

3 个答案:

答案 0 :(得分:12)

你可以用反射来做到这一点:

public void CopyNonNullProperties(object source, object target)
{
    // You could potentially relax this, e.g. making sure that the
    // target was a subtype of the source.
    if (source.GetType() != target.GetType())
    {
        throw new ArgumentException("Objects must be of the same type");
    }

    foreach (var prop in source.GetType()
                               .GetProperties(BindingFlags.Instance |
                                              BindingFlags.Public)
                               .Where(p => !p.GetIndexParameters().Any())
                               .Where(p => p.CanRead && p.CanWrite))
    {
        var value = prop.GetValue(source, null);
        if (value != null)
        {
            prop.SetValue(target, value, null);
        }
    }
}

答案 1 :(得分:2)

从你的例子来看,我认为你正在寻找这样的事情:

static void CopyTo<T>(T from, T to)
{
    foreach (PropertyInfo property in typeof(T).GetProperties())
    {
        if (!property.CanRead || !property.CanWrite || (property.GetIndexParameters().Length > 0))
            continue;

        object value = property.GetValue(to, null);
        if (value != null)
            property.SetValue(to, property.GetValue(from, null), null);
    }
}

答案 2 :(得分:2)

如果您打算多次使用,可以使用已编译的表达式来获得更好的性能:

public static class Mapper<T>
{
    static Mapper()
    {
        var from = Expression.Parameter(typeof(T), "from");
        var to = Expression.Parameter(typeof(T), "to");

        var setExpressions = typeof(T)
            .GetProperties()
            .Where(property => property.CanRead && property.CanWrite && !property.GetIndexParameters().Any())
            .Select(property =>
            {
                var getExpression = Expression.Call(from, property.GetGetMethod());
                var setExpression = Expression.Call(to, property.GetSetMethod(), getExpression);
                var equalExpression = Expression.Equal(Expression.Convert(getExpression, typeof(object)), Expression.Constant(null));

                return Expression.IfThen(Expression.Not(equalExpression), setExpression);
            });

        Map = Expression.Lambda<Action<T, T>>(Expression.Block(setExpressions), from, to).Compile();
    }

    public static Action<T, T> Map { get; private set; }
}

并像这样使用它:

Mapper<Entity>.Map(e1, e2);