将C#LINQ表达式用于值类型和引用类型

Lan*_*don 3 c# linq generics dynamic c#-4.0

我正在使用MVC for REST,以便我可以利用Razor输出不同类型.CSV是这些输出之一.而不是为每种输入类型编写此模板:

ID,Created,Content
@foreach (var item in Model.TimeData)
{
<text>@item.ID,@item.Created,"@Html.Raw(item.Content.Replace("\"", "\"\""))"</text>
}
Run Code Online (Sandbox Code Playgroud)

我想利用paramsSystem.Linq.Expressions.Expression编写这样的东西:

@{
    Html.WriteCsv<TimeObject>(Model.TimeData, p => p.ID, p => p.Created, p => p.Content);   
}
Run Code Online (Sandbox Code Playgroud)

我开始编写泛型HtmlHelper并很快意识到我遇到了值类型的问题(memberExpression将为null).下面的代码试图写出CSV标题(ID,Created,Content),但它只输出"Content"(因为ID和Created是值类型(intDateTime)).

public static void WriteCsv<TModel>(this HtmlHelper htmlHelper, List<TModel> list, params Expression<Func<TModel, object>>[] expressions)
{
    foreach (var expression in expressions)
    {
        MemberExpression memberExpression = expression.Body as MemberExpression;

        if (memberExpression != null)
        {
            var propertyInfo = (PropertyInfo)memberExpression.Member;

            htmlHelper.ViewContext.Writer.Write(propertyInfo.Name + Environment.NewLine);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

我试图替换objectdynamic,以为会工作,但是当我快看expression.Body,它似乎仍然认为它在处理一个对象(DebugView属性(System.Object)$p.ID).

这在C#4.0中是不可能的吗?

这是我正在使用它的类型:

[DataContract(IsReference = true, Namespace = "urn:test:TimeObject")]
public class TimeObject
{
    [DataMember]
    public long ID { get; set; }

    [DataMember]
    public string Content { get; set; }

    [DataMember]
    public DateTime Created { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

Pau*_*aft 6

在表达式引用值类型的情况下,编译器必须使用引用; 它含蓄地这样做.这种复杂性意味着Value Type成员表达式的表达式树不仅仅是MemberExpression,因此您的强制转换返回null.

以下是从"值类型"或"引用类型"成员表达式获取属性名称的一般解决方案,取自此问题:

private string GetPropertyName(Expression<Func<object, object>> f) {
    var body = f.Body;
    if (body.NodeType==ExpressionType.Convert)
      body = ((UnaryExpression) body).Operand;
    if ((body as MemberExpression) != null) {
        return (body as MemberExpression).Member.Name;
    }
    return "";
}
Run Code Online (Sandbox Code Playgroud)