2012-04-17 2 views
7

В моем проекте есть отдельные модульные тесты, в которых мы хотим установить некоторые свойства, которые имеют частные сеттеры. В настоящее время я делаю это с помощью отражения, и этот метод расширения:Как передать свойство через выражение лямбда?

public static void SetPrivateProperty(this object sourceObject, string propertyName, object propertyValue) 
{ 
    sourceObject.GetType().GetProperty(propertyName).SetValue(sourceObject, propertyValue, null); 
} 

Если предположить, что у меня был TestObject так:

public class TestObject 
{ 
    public int TestProperty{ get; private set; } 
} 

Затем я могу назвать это в моих модульных тестов следующим образом:

myTestObject.SetPrivateProperty("TestProperty", 1); 

Тем не менее, я хотел бы иметь подтверждение имени свойства во время компиляции, и поэтому я хотел бы передать свойство в выражении через выражение, например:

myTestObject.SetPrivateProperty(o => o.TestProperty, 1); 

Как это сделать?

+0

Какова цель лямбда-выражения? Обеспечить проверку времени компиляции? – mellamokb

+0

@mellamokb Да. Если есть еще одно средство, я играю. – Sterno

+0

См. Http://stackoverflow.com/questions/671968/retrieving-property-name-from-lambda-expression – phoog

ответ

9

Если геттер является общедоступным, то следующее должно работать. Он даст вам метод расширения, который выглядит следующим образом:

var propertyName = myTestObject.NameOf(o => o.TestProperty); 

Для этого требуется публичный геттер. Надеюсь, в какой-то момент функциональность отражения, подобная этой, перевернута на язык.

public static class Name 
{ 
    public static string Of(LambdaExpression selector) 
    { 
     if (selector == null) throw new ArgumentNullException("selector"); 

     var mexp = selector.Body as MemberExpression; 
     if (mexp == null) 
     { 
      var uexp = (selector.Body as UnaryExpression); 
      if (uexp == null) 
       throw new TargetException(
        "Cannot determine the name of a member using an expression because the expression provided cannot be converted to a '" + 
        typeof(UnaryExpression).Name + "'." 
       ); 
      mexp = uexp.Operand as MemberExpression; 
     } 

     if (mexp == null) throw new TargetException(
      "Cannot determine the name of a member using an expression because the expression provided cannot be converted to a '" + 
      typeof(MemberExpression).Name + "'." 
     ); 
     return mexp.Member.Name; 
    } 

    public static string Of<TSource>(Expression<Func<TSource, object>> selector) 
    { 
     return Of<TSource, object>(selector); 
    } 

    public static string Of<TSource, TResult>(Expression<Func<TSource, TResult>> selector) 
    { 
     return Of(selector as LambdaExpression); 
    } 
} 

public static class NameExtensions 
{ 
    public static string NameOf<TSource, TResult>(this TSource obj, Expression<Func<TSource, TResult>> selector) 
    { 
     return Name.Of(selector); 
    } 
}