Expression Tree 扩展MVC中的 HtmlHelper 和 UrlHelper

表达式树是LINQ To everything 的基础,同时各种类库的Fluent API也 大量使用了Expression Tree。还记得我在不懂expression tree时,各种眼花缭乱的API 看的我各种膜拜,当我熟悉expression tree 后恍然大悟,不用看代码也能知道别人的API 是如何设计的(^_^)。 接下来这篇博客就谈谈如何使用expression tree扩展MVC中的HtmlHelperUrlHelper

场景

当我们在MVC中生成一个action的url会这样写:var url=UrlHelper.Action("index", "Home"); 如果要render一个action时会这样写:Html.RenderAction("index", "Home");

这样的写法瑕疵在于我们传递了两个字符串类型的参数在代码中,而我们又免不了对action和controller做重命名操作:index->default, 即便是你用resharper这样的工具重命名也无法将UrlHelper.Action("index", "Home"); 改变为UrlHelper.Action("default", "Home");

vs甚至在编译时都不会检查出来这个错误。 所以我们的目标是:设计出具有静态检查的API,让vs提示出这个错误来,甚至是重命名时直接把相关代码都能重命名。

使用Expression Tree 重新设计这两组API

目标:设计出类似的API:Url.Action((HomeController c) => c.Index());

1.很明显我们需要在UrlHelper上写个扩展方法:

public static string Action<TController>(
    this UrlHelper url,
    Expression<Func<TController, ActionResult>> actionSelector,
    string protocol = null,
    string hostname = null)
    {
        var action = "Index"; //待解析
        var controller = "Home";//带解析
        var routeValues = new RouteValueDictionary();//待解析

        return url.Action(action, controller, routeValues, protocol, hostname);
    }

现在只需要根据表达式Expression<Func<TController, ActionResult>> actionSelector 解析出action,controller,还有routeValues即可

2.解析出controller 的名称

分析:controller的名称可以根据泛型方法中的泛型参数TController得到

private static string GetControllerName(Type controllerType)
{
    var controllerName = controllerType.Name.EndsWith("Controller")
        ? controllerType.Name.Substring(0, controllerType.Name.Length - "Controller".Length)
        : controllerType.Name;
    return controllerName;
}

3.解析action的名称

分析:由于表达式Expression<Func<TController, ActionResult>> actionSelector 是一个MethodCallExpression, 所以可以很容易得到action的名称:var action=call.Method.Name;

这样已经完成了最简单的url构造方案, 但是我们还没有处理带有参数的action类型,例如:Url.Action((HomeController c) => c.Detail(10, 20));

4.解析routeValues

分析:action中的参数实际上就是MethodCallExpression中的参数,我们解析这个expression的参数即可,然后得到RouteValues

private static RouteValueDictionary GetRouteValues(MethodCallExpression call)
{
    var routeValues = new RouteValueDictionary();

    var args = call.Arguments;
    ParameterInfo[] parameters = call.Method.GetParameters();
    var pairs = args.Select((a, i) => new
    {
        Argument = a,
        ParamName = parameters[i].Name
    });
    foreach (var argumentParameterPair in pairs)
    {
        string name = argumentParameterPair.ParamName;
        object value = argumentParameterPair.Argument.GetValue();
        if (value != null)
        {
            var valueType = value.GetType();
            if (valueType.IsValueType)
            {
                routeValues.Add(name, value);
            }
            throw new NotSupportedException("unsoupported parameter type {0}".FormatWith(value.ToString()));
        }
    }
    return routeValues;
}

如此一来,类似Url.Action((HomeController c) => c.Detail(10, 20));这样的action也可以构造出Url了。

if (valueType.IsValueType)
 {
     routeValues.Add(name, value);
 }

此代码并不支持复杂类型的参数,对于action中传入复杂的类型,比如:

public class User
{
    public int Age { get; set; }
    public string Email { get; set; }
}

如果Action中的参数使用了User类型:

public ActionResult SayHelloToUser(User user)
{
    return new EmptyResult();
}

如何解析呢?

var properties = PropertyInfoHelper.GetProperties(valueType);
foreach (var propertyInfo in properties)
  {
     routeValues.Add(propertyInfo.Name, propertyInfo.GetValue(value));
  }

大功告成,现在已经完美解决了各种类型的参数传入。

同样的道理,我们可以扩展HtmlHelperRenderAction(), ActionLink()….

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值