Expression tree for a member access of depth > 1

2019-04-17 23:54发布

问题:

public class Job
{
    public string Name { get; set; }
    public int Salary { get; set; }
}
public class Employee
{
    public string Name { get; set; }
    public Job Job { get; set; }
}

If I want to create an expression tree of a member access to Employee.Name this is what I do:

        var param = Expression.Parameter(type, "x");
        var memberAccess = Expression.PropertyOrField(param, memberName);
        return Expression.Lambda<Func<TModel, TMember>>(memberAccess, param);

What is the equivalent to this for a member access to Employee.Job.Salary ?

回答1:

You need:

var jobProperty = Expression.PropertyOrField(param, "Job");
var salaryProperty = Expression.PropertyOrField(jobProperty, "Salary");

Basically you're taking the Salary property from the result of evaluating x.Job.

If you need to do this in a programmatic way, you'll need something like:

Expression expression = Expression.Parameter(type, "x");
foreach (var property in properties.Split('.'))
{
    expression = Expression.PropertyOrField(expression, property);
}


回答2:

The best way will be create Extension as here:

public static class ExpressionExtensions
{
    /// <summary>
    ///     create expression by property name
    /// </summary>
    /// <typeparam name="TModel"></typeparam>
    /// <param name="propertyName">
    ///     <example>Urer.Role.Name</example>
    /// </param>
    /// <returns></returns>
    public static Expression<Func<TModel, dynamic>> CreateExpression<TModel>(this string propertyName) {
        Type currentType = typeof (TModel);
        ParameterExpression parameter = Expression.Parameter(currentType, "x");
        Expression expression = parameter;

        int i = 0;
        List<string> propertyChain = propertyName.Split('.').ToList();
        do {
            System.Reflection.PropertyInfo propertyInfo = currentType.GetProperty(propertyChain[i]);
            currentType = propertyInfo.PropertyType;
            i++;
            if (propertyChain.Count == i)
            {
                currentType = typeof (object);
            }
            expression = Expression.Convert(Expression.PropertyOrField(expression, propertyInfo.Name), currentType);
        } while (propertyChain.Count > i);

        return Expression.Lambda<Func<TModel, dynamic>>(expression, parameter);
    }
}

You cannot Convert() to typeof(object) everytime, because System.Object doesn't have property, that your type has (like Name or Salary in you example).