如何使用表达式树对数值类型应用隐式强制转换?

时间:2015-08-15 11:00:10

标签: c# dynamic expression-trees expandoobject

我有一个带有int字段的ExpandoObject,我想使用表达式树将其强制转换为十进制。

以下是我使用的方法:

private static Expression<Func<dynamic, decimal>> CreateLambdaCastExpression()
    {
        // source
        var sourceParameterExpression = Expression.Parameter(typeof (object), "source");

        var binder = Binder.GetMember(
            CSharpBinderFlags.None, "IntProp", typeof (ExpressionTreeUtils),
            new[] {CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null)});
        // source.sourceProperty
        var sourcePropertyExpression = Expression.Dynamic(
            binder, typeof (object), sourceParameterExpression);

        // (decimal) source;
        var castedValueExpression = Expression.Convert(sourcePropertyExpression, typeof (decimal));

        // () =>  (decimal) source;
        return Expression.Lambda<Func<dynamic, decimal>>(castedValueExpression,
            sourceParameterExpression);
    }

以这种方式调用会导致InvalidCastException:

        dynamic source = new ExpandoObject();
        source.IntProp = 1;

        decimal r = CreateLambdaCastExpression().Compile()(source);

如果我将source.IntProp设置为1m,它可以(显然)

我已在msdn上读到ExpressionConvert仅对用户定义的类型执行隐式转换,因此可能就是解释。

关于如何对数字类型执行隐式转换的任何想法?

1 个答案:

答案 0 :(得分:3)

更改这些行:

// CSharpBinderFlags.ConvertExplicit: explicit cast 
// (will convert double to decimal)
// CSharpBinderFlags.None: implicit cast
// (will convert int to decimal, won't convert double to decimal)
var convert = Binder.Convert(CSharpBinderFlags.ConvertExplicit, typeof(decimal), typeof(ExpressionTreeUtils));

// (decimal) source;
var castedValueExpression = Expression.Dynamic(
    convert, typeof(decimal), sourcePropertyExpression);

请注意有关隐式/显式转换的注释。

(所以唯一改变的是var castedValueExpression的构建方式)

ideone完整示例。