Espressione > come posso ottenere il nome di TValue?

.net c# expression-trees

Domanda

//ModelFor(person =>person.Name);
public void ModelFor<TModel, TValue>(
    Expression<Func<TModel, TValue>> expression)
{
    //Result should be "Name"
    string nameOfTValue = ????;     
}

Risposta accettata

EDIT : dopo la modifica, penso che tu voglia il nome del membro coinvolto nell'espressione, assumendo naturalmente che l'espressione sia un'espressione membro in primo luogo.

((MemberExpression)expression.Body).Member.Name

Per essere più robusti, puoi fare:

var memberEx = expression.Body as MemberExpression;

if (memberEx == null)
     throw new ArgumentException("Body not a member-expression.");

string name = memberEx.Member.Name;

(Non più pertinente):

Per ottenere un System.Type che rappresenta il tipo di argomento del tipo TValue , è possibile utilizzare l'operatore typeof .

Probabilmente vuoi:

typeof(TValue).Name

Ma considera anche le proprietà FullName e AssemblyQualifiedName , se appropriato.

Questo in realtà non ha nulla a che fare con gli alberi di espressione; è possibile utilizzare questa tecnica per ottenere il tipo di argomento di tipo per qualsiasi metodo generico.


Risposta popolare

@Ani: Non penso che sia giusto, penso che voglia il nome del parametro nell'espressione di tipo TValue

Se questo è vero ... funziona solo a un livello di profondità ma potrebbe comunque essere utile:

var nameOfTValue = ((MemberExpression)expression.Body).Member.Name; 

Ecco un'implementazione più intelligente che dovrebbe essere in grado di gestire più livelli:

 public class PropertyName{
    public static string For<T>(
      Expression<Func<T,object>> expression){
      var body=expression.Body;
      return GetMemberName(body);
    }
    public static string For(
      Expression<Func<object>> expression){
      var body=expression.Body;
      return GetMemberName(body);
    }
    public static string GetMemberName(
      Expression expression){
      if(expression is MemberExpression){
        var memberExpression=(MemberExpression)expression;
        if(memberExpression.Expression.NodeType==
           ExpressionType.MemberAccess)
          return GetMemberName(memberExpression.Expression)
            +"."+memberExpression.Member.Name;
        return memberExpression.Member.Name;
      }
      if(expression is UnaryExpression){
        var unaryExpression=(UnaryExpression)expression;
        if(unaryExpression.NodeType!=ExpressionType.Convert)
          throw new Exception(string.Format
            ("Cannot interpret member from {0}",expression));
        return GetMemberName(unaryExpression.Operand);
      }
      throw new Exception
        (string.Format("Could not determine member from {0}",expression));
    }
  }

Uso:

var fieldName=PropertyName.For<Customer>(x=>x.Address.Region);
//fieldName==Address.Region

Un altro trucco, questo può essere combinato bene con la riflessione:

public static T Set<T,TProp>(this T o,
   Expression<Func<T,TProp>> field,TProp value){
  var fn=((MemberExpression)field.Body).Member.Name;
  o.GetType().GetProperty(fn).SetValue(o,value,null);
  return o;
}

Permette di impostare direttamente le proprietà con facilità, può essere utile per i dispositivi di prova:

var customer=new Customer("firstName","lastName");
customer.Set(x=>x.Name, "different firstName");


Autorizzato sotto: CC-BY-SA with attribution
Non affiliato con Stack Overflow
È legale questo KB? Sì, impara il perché
Autorizzato sotto: CC-BY-SA with attribution
Non affiliato con Stack Overflow
È legale questo KB? Sì, impara il perché