具有动态参数的表达式树

Mat*_*ero 8 .net c# dynamic expression-trees c#-4.0

我想转换这个:

Func<dynamic, object> myFunc = t => return t.Name + " " + t.Surname;
Run Code Online (Sandbox Code Playgroud)

进入表达式树.

我想出的是:

ParameterExpression target = ExpressionParameter(typeof(dynamic), "target");
ParameterExpression result = ExpressionParameter(typeof(object), "result");
BlockExpression block = Expression.Block(
     new [] { result },
     Expression.Assign(
           result,
           Expression.Add(
                 Expression.Add(
                      Expression.Property(target, "Name"),
                      Expression.Constant(" ", typeof(string))
                 ),
                 Expression.Property(target, "Surname")
           )
     )
);
Func<dynamic, object> myFunc = Expression.Lambda<dynamic, object>>(block, target).Compile();
Run Code Online (Sandbox Code Playgroud)

但是,编译器不喜欢typeof(dynamic),我有点喜欢它.dynamic不是一种类型,它object本质上是一种类型.

所以我继续改变ParameterExpression:

ParameterExpression target = ExpressionParameter(typeof(object), "target");
Run Code Online (Sandbox Code Playgroud)

代码现在编译,但在运行时存在问题.

我试图让财产的价值Nametarget,这可能有道理的,如果对象是dynamic.

但由于target被认为是类型object,因此Expression会抛出一个错误,告诉我Name不存在属性.

是否有用于获取动态属性的Expression?

Mat*_*ero 5

对于那些对解决方案感兴趣或感兴趣的人:

ParameterExpression target = Expression.Parameter(typeof(object), "target");
ParameterExpression result = Expression.Parameter(typeof(object), "result");

CallSiteBinder getName = Binder.GetMember(
   CSharpBinderFlags.None, "Name", typeof(Program),
   new CSharpArgumentInfo[] {
       CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null)
   }
);

CallSiteBinder getSurname= Binder.GetMember(
   CSharpBinderFlags.None, "Surname", typeof(Program),
   new CSharpArgumentInfo[] {
       CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null)
   }
);

BlockExpression block = Expression.Block(
    new[] { result },
    Expression.Assign(
        result,
        Expression.Call(typeof(string).GetMethod("Concat", new Type[] { typeof(object[]) }),
                        Expression.NewArrayInit(typeof(object),
                             Expression.Dynamic(getName, typeof(object), target),
                             Expression.Constant(" ", typeof(object)),
                             Expression.Dynamic(getSurname, typeof(object), target)
                        )
       )
    )
);

Func<dynamic, object> myFunc = Expression.Lambda<Func<dynamic, object>>(block, target).Compile();
Run Code Online (Sandbox Code Playgroud)

这正是我在做什么:

  1. 创建了一个CallSiteBinder获取Name作为参数传递的动态对象的动态属性的值
  2. 创建了一个CallSiteBinder获取Surname作为参数传递的动态对象的动态属性的值
  3. 调用了方法string.Concat(params object[] args)。为此,我需要将我的参数作为object. 我创建与该数组getName" "getSurname

我使用以下答案作为指导和参考:

C# 4 表达式树中的“动态”

使用上述方法,可以执行以下操作:

dynamic person = new ExpandoObject();
person.Name = "Matt";
person.Surname = "Smith";

object value = myFunc(person);
Console.WriteLine(value); //Will print out "Matt Smith"

//Internally it just calls:
//string.Concat(new object[] { person.Name, " ", person.Surname });
Run Code Online (Sandbox Code Playgroud)