2012-04-17 8 views
7

Existen algunas pruebas unitarias en mi proyecto en las que queremos poder establecer algunas propiedades que tienen entidades privadas. Actualmente lo estoy haciendo a través de la reflexión y la extensión de este método:¿Cómo puedo pasar una propiedad a través de una expresión lambda?

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

Suponiendo que tenía un TestObject así:

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

puedo luego llamar a esto en mis pruebas de unidad de la siguiente manera:

myTestObject.SetPrivateProperty("TestProperty", 1); 

Sin embargo, me gustaría tener la validación del nombre de la propiedad en tiempo de compilación, y así me gustaría poder pasar la propiedad en la expresión, de esta manera:

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

¿Cómo puedo hacer esto?

+0

¿Cuál es el propósito de la expresión lambda? Para proporcionar validación en tiempo de compilación? – mellamokb

+0

@mellamokb Sí. Si hay otro medio para hacer eso, soy un juego. – Sterno

+0

Ver http://stackoverflow.com/questions/671968/retrieving-property-name-from-lambda-expression – phoog

Respuesta

9

Si el captador es público, entonces lo siguiente debería funcionar. Le dará un método de extensión que se ve así:

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

Requiere un getter público. Espero que, en algún momento, la funcionalidad de reflexión como esta se desarrolle en el lenguaje.

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); 
    } 
} 
Cuestiones relacionadas