How to create ExpressionTree with multiple method calls

I currently have the following code that allows me to call any method required on a property of EmailAddress

my object and it works great:

public static Expression<Func<T, bool>> BuildEmailAddressLambda(string method, params object[] args) {
    var e = Expression.Parameter(typeof(T), "e");
    var propertyInfo = typeof(T).GetProperty("EmailAddress");
    var m = Expression.MakeMemberAccess(e, propertyInfo);
    var mi = m.Type.GetMethod(method, args.Select(a => a.GetType()).ToArray());
    var c = args.Select(a => Expression.Constant(a, a.GetType())).ToArray();

    Expression<Func<T, bool>> lambda = Expression.Lambda<Func<T, bool>>(Expression.Call(m, mi, c), e);
    return lambda;
}

// called:
lambda = LambdaExpressionHelper<MailingListMember>.BuildEmailAddressLambda("StartsWith", "r", StringComparison.OrdinalIgnoreCase);

      

However, now I need to make changes to the code to make it work for members of my object except EmailAddress

. Specifically, I would like to create a tree to cover the following expression, which uses multiple method calls:

e.GetStringValue(12).StartsWith("r", StringComparison.OrdinalIgnoreCase); 

      

I have made several attempts, all of which end up with various errors. I feel like I am missing something in the logic for creating ExpressionTrees, and with this help, you will be very grateful.

thank

+3


source to share


1 answer


Is this good for you?



public static Expression<Func<T, bool>> BuildEmailAddressLambda<T>(
    string member, IEnumerable<object> memberArgs, string method, params object[] args)
{
    var e = Expression.Parameter(typeof(T), "e");
    var memberInfo =
        (MemberInfo) typeof(T).GetField(member) ??
        (MemberInfo) typeof(T).GetProperty(member) ??
        (MemberInfo) typeof(T).GetMethod(member, (memberArgs ?? Enumerable.Empty<object>()).Select(p => p.GetType()).ToArray());
    Expression m;
    if (memberInfo.MemberType == MemberTypes.Method)
    {
        var a = memberArgs.Select(p => Expression.Constant(p));
        m = Expression.Call(e, (MethodInfo) memberInfo, a);
    }
    else
    {
        m = Expression.MakeMemberAccess(e, memberInfo);
    }
    var mi = m.Type.GetMethod(method, args.Select(a => a.GetType()).ToArray());
    var c = args.Select(a => Expression.Constant(a, a.GetType()));

    return Expression.Lambda<Func<T, bool>>(Expression.Call(m, mi, c), e);
}

// called:
lambda = LambdaExpressionHelper<MailingListMember>.BuildEmailAddressLambda("EmailAddress", null, "StartsWith", "r", StringComparison.OrdinalIgnoreCase);
// or
lambda = LambdaExpressionHelper<MailingListMember>.BuildEmailAddressLambda("GetStringValue", new object[] { 12 }, "StartsWith", "r", StringComparison.OrdinalIgnoreCase);

      

+3


source







All Articles