如何创建一个接受多个lambda表达式作为参数的方法?

时间:2013-02-14 17:02:05

标签: c# asp.net-mvc linq lambda

我正在尝试创建自己的Extension方法,该方法可以使用任意数量的lambda表达式,但是当我添加多个表达式时,它似乎会窒息。

以下是方法:

public static MvcHtmlString _RouteButton<TModel, TProperty>(this HtmlHelper<TModel> htmlHelper, string label, string controller, string action, params Expression<Func<TModel, TProperty>>[] parameters)
{
   var test = parameters;
   return MvcHtmlString.Empty;
}

以下是成功调用它的标记:

<%: Html._RouteButton("details", "Health", "SystemDetails", m=>m.Id)%>

以下是错误标记:

<%: Html._RouteButton("details", "Health", "SystemDetails", m=>m.Id, m=>m.Status)%>

这是错误:

  

无法从用法推断出方法的类型参数。尝试   明确指定类型参数

感谢任何帮助。谢谢!

4 个答案:

答案 0 :(得分:8)

让我们简化:

using System;
class P
{
    static void M<R>(params Func<R>[] p) {}
    static void N(int i, string s, decimal m)
    {
        M(()=>i, ()=>s); // fails
        M(()=>i, ()=>s.Length); // succeeds
        M(()=>i, ()=>m); // succeeds
    }
}

现在很明显为什么你的程序失败了?

在我的程序中,每次调用都会尝试推断R.在第一次调用中,R被推断为int和string,因此推理失败,因为没有int和string类型。在第二个,R被推断为int和... int再次!这是成功的,因为int匹配所有边界。在第三个中,R被推断为int和decimal,这是成功的,因为每个int都可以隐式转换为十进制,因此十进制是一个很好的推理。

Id和Status可能是属性不兼容的属性。如果您想这样做,那么推断的其中一种类型必须是最佳类型

请注意,C#从不说“哦,我看到你推断出狗,猫和鱼的界限,所以我猜你的意思是动物”。 C#反而说“狗,猫或鱼都不是最好的约束,所以我不知道你的意思;请明确说明”。

答案 1 :(得分:4)

Eric Lippert的回答解释了问题所在。我将添加如何实际解决它:您很可能不需要lambda来返回属性的类型,您只是想获取表达式树来检查属性。因此,您可以做的是将类型从Func<TModel, TProperty>更改为Func<TModel, object>(并删除TProperty类型参数)。由于C#中的所有普通类型都可以隐式转换为object,因此您的代码将通过此更改进行编译并正常运行

答案 2 :(得分:1)

我想在网站上看到这个,以防有人尝试类似的东西。

代码构建路径表单。要执行此操作,需要属性的名称及其

public static void _RouteButton<TModel>(this HtmlHelper<TModel> htmlHelper, string text, string controller, string action, params Expression<Func<TModel, object>>[] parameters)
{
    using (htmlHelper.BeginRouteForm("Default", new { controller = controller, action = action }))
    {
        foreach (Expression<Func<TModel, object>> p in parameters)
        {
            MemberExpression me;
            switch (p.Body.NodeType)
            {
                case ExpressionType.Convert:
                case ExpressionType.ConvertChecked:
                    var ue = p.Body as UnaryExpression;
                    me = ((ue != null) ? ue.Operand : null) as MemberExpression;
                    break;
                default:
                    me = p.Body as MemberExpression;
                    break;
            }
            string name = me.Member.Name;
            string value = p.Compile()(htmlHelper.ViewData.Model).ToString();
            HttpContext.Current.Response.Write(htmlHelper.Hidden(name, value).ToHtmlString());
        }
        HttpContext.Current.Response.Write("<input type='submit' value='" + text + "' />");
    }
}

答案 3 :(得分:-1)

@ eric-lippert根据你的例子解释“为什么它不起作用”。每个表达式中传递的属性类型都不相同。

将扩展方法的签名更改为仅指定TModel,因为通用将有助于解决问题。另外,为表达式函数指定对象代替TProperty。

扩展方法

public static MvcHtmlString _RouteButton<TModel>(this HtmlHelper<TModel> htmlHelper, 
                                                 params Expression<Func<TModel, object>>[] parameters)
{
    var test = parameters;
    return MvcHtmlString.Empty;
}

<强> ASP /剃须刀

<%: Html._RouteButton(m=>m.Id, m=>m.Status)%>
@Html._RouteButton(m=>m.Id, m=>m.Status)