首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Expression<Func<TModel,string>> to Expression<Action<TModel>> "Getter“to "Setter”

Expression<Func<TModel,string>> to Expression<Action<TModel>> "Getter“to "Setter”
EN

Stack Overflow用户
提问于 2011-10-11 17:10:54
回答 5查看 3.9K关注 0票数 9

我是表达式的新手,我想知道如何以任何方式转换我的表达式

假设在本例中,我的TModel是Customer类型,并将其分配到如下位置:

代码语言:javascript
复制
Expression<Func<TModel, string>> getvalueexpression = customer =>customer.Name

类似这样的东西

代码语言:javascript
复制
Expression<Action<TModel,string>> setvalueexpression = [PSEUDOCODE] getvalueexpression = input
Action<TModel,string> Setter  = setvalueexpression.Compile();
Setter(mycustomer,value);

因此,简而言之,我希望以某种方式构建并编译一个表达式,将我的getter表达式指定的客户名称设置为一个特定值。

EN

回答 5

Stack Overflow用户

回答已采纳

发布于 2011-10-11 17:28:54

修改后的版本。这个类可能比你能找到的许多其他类更好:-)这是因为这个版本支持直接属性(p => p.B) (和其他人一样:- ),嵌套属性(p => p.B.C.D),字段(“终端”和“中间”,所以在p => p.B.C.D中,BD都可以是字段)和类型的“内部”转换(所以p => ((BType)p.B).C.Dp => (p.B as BType).C.D) )。唯一不支持的就是“终端”元素的类型转换(所以没有p => (object)p.B)。

生成器中有两个“代码路径”:用于简单表达式(p => p.B)和用于“嵌套”表达式。存在.NET 4.0 (具有Expression.Assign表达式类型)的代码变体。从我的一些基准测试中,最快的委托是:属性的“简单”Delegate.CreateDelegate,字段的Expression.Assign和字段的“简单”FieldSetter (这个只比字段的Expression.Assign慢一点)。所以在.NET 4.0下,你应该去掉所有标记为3.5的代码。

代码的一部分不是我的。最初的(简单)版本是基于流畅的NHibernate代码(但它只支持直接属性),其他一些部分是基于How do I set a field value in an C# Expression tree?Assignment in .NET 3.5 expression trees的代码。

代码语言:javascript
复制
public static class FluentTools
{
    public static Action<T, TValue> GetterToSetter<T, TValue>(Expression<Func<T, TValue>> getter)
    {
        ParameterExpression parameter;
        Expression instance;
        MemberExpression propertyOrField;

        GetMemberExpression(getter, out parameter, out instance, out propertyOrField);

        // Very simple case: p => p.Property or p => p.Field
        if (parameter == instance)
        {
            if (propertyOrField.Member.MemberType == MemberTypes.Property)
            {
                // This is FASTER than Expression trees! (5x on my benchmarks) but works only on properties
                PropertyInfo property = propertyOrField.Member as PropertyInfo;
                MethodInfo setter = property.GetSetMethod();
                var action = (Action<T, TValue>)Delegate.CreateDelegate(typeof(Action<T, TValue>), setter);
                return action;
            }
            #region .NET 3.5
            else // if (propertyOrField.Member.MemberType == MemberTypes.Field)
            {
                // 1.2x slower than 4.0 method, 5x faster than 3.5 method
                FieldInfo field = propertyOrField.Member as FieldInfo;
                var action = FieldSetter<T, TValue>(field);
                return action;
            }
            #endregion
        }

        ParameterExpression value = Expression.Parameter(typeof(TValue), "val");

        Expression expr = null;

        #region .NET 3.5
        if (propertyOrField.Member.MemberType == MemberTypes.Property)
        {
            PropertyInfo property = propertyOrField.Member as PropertyInfo;
            MethodInfo setter = property.GetSetMethod();
            expr = Expression.Call(instance, setter, value);
        }
        else // if (propertyOrField.Member.MemberType == MemberTypes.Field)
        {
            expr = FieldSetter(propertyOrField, value);
        }
        #endregion

        //#region .NET 4.0
        //// For field access it's 5x faster than the 3.5 method and 1.2x than "simple" method. For property access nearly same speed (1.1x faster).
        //expr = Expression.Assign(propertyOrField, value);
        //#endregion

        return Expression.Lambda<Action<T, TValue>>(expr, parameter, value).Compile();
    }

    private static void GetMemberExpression<T, U>(Expression<Func<T, U>> expression, out ParameterExpression parameter, out Expression instance, out MemberExpression propertyOrField)
    {
        Expression current = expression.Body;

        while (current.NodeType == ExpressionType.Convert || current.NodeType == ExpressionType.TypeAs)
        {
            current = (current as UnaryExpression).Operand;
        }

        if (current.NodeType != ExpressionType.MemberAccess)
        {
            throw new ArgumentException();
        }

        propertyOrField = current as MemberExpression;
        current = propertyOrField.Expression;

        instance = current;

        while (current.NodeType != ExpressionType.Parameter)
        {
            if (current.NodeType == ExpressionType.Convert || current.NodeType == ExpressionType.TypeAs)
            {
                current = (current as UnaryExpression).Operand;
            }
            else if (current.NodeType == ExpressionType.MemberAccess)
            {
                current = (current as MemberExpression).Expression;
            }
            else
            {
                throw new ArgumentException();
            }
        }

        parameter = current as ParameterExpression;
    }

    #region .NET 3.5

    // Based on https://stackoverflow.com/questions/321650/how-do-i-set-a-field-value-in-an-c-expression-tree/321686#321686
    private static Action<T, TValue> FieldSetter<T, TValue>(FieldInfo field)
    {
        DynamicMethod m = new DynamicMethod("setter", typeof(void), new Type[] { typeof(T), typeof(TValue) }, typeof(FluentTools));
        ILGenerator cg = m.GetILGenerator();

        // arg0.<field> = arg1
        cg.Emit(OpCodes.Ldarg_0);
        cg.Emit(OpCodes.Ldarg_1);
        cg.Emit(OpCodes.Stfld, field);
        cg.Emit(OpCodes.Ret);

        return (Action<T, TValue>)m.CreateDelegate(typeof(Action<T, TValue>));
    }

    // Based on https://stackoverflow.com/questions/208969/assignment-in-net-3-5-expression-trees/3972359#3972359
    private static Expression FieldSetter(Expression left, Expression right)
    {
        return
            Expression.Call(
                null,
                typeof(FluentTools)
                    .GetMethod("AssignTo", BindingFlags.NonPublic | BindingFlags.Static)
                    .MakeGenericMethod(left.Type),
                left,
                right);
    }

    private static void AssignTo<T>(ref T left, T right)  // note the 'ref', which is
    {                                                     // important when assigning
        left = right;                                     // to value types!
    }

    #endregion
}
票数 11
EN

Stack Overflow用户

发布于 2011-10-11 18:07:21

代码语言:javascript
复制
static Expression<Action<T, TProperty>> MakeSetter<T, TProperty>(Expression<Func<T, TProperty>> getter)
{
    var memberExpr = (MemberExpression)getter.Body;
    var @this = Expression.Parameter(typeof(T), "$this");
    var value = Expression.Parameter(typeof(TProperty), "value");
    return Expression.Lambda<Action<T, TProperty>>(
        Expression.Assign(Expression.MakeMemberAccess(@this, memberExpr.Member), value),
        @this, value);
}
票数 3
EN

Stack Overflow用户

发布于 2011-10-11 17:22:40

我有这个辅助方法,它返回一个属性的属性信息:

代码语言:javascript
复制
public static PropertyInfo GetPropertyInfo<T, U>(Expression<Func<T, U>> property) where T : class
{
    var memberExpression = (property.Body as MemberExpression);

    if (memberExpression != null && memberExpression.Member is PropertyInfo)
    {
        return memberExpression.Member as PropertyInfo;
    }

    throw new InvalidOperationException("Invalid usage of GetPropertyInfo");
}

用法:GetPropertyInfo((MyClass c) => c.PropertyName);

然后,可以使用PropertyInfo设置类的属性值。

您需要修改代码以满足您的需要,但希望这会有所帮助。

票数 2
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/7723744

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档