代码改变世界

使用表达式树构建DomainRoute的URL

2009-08-31 15:48 Jeffrey Zhao 阅读(...) 评论(...) 编辑 收藏

由于DomainRoute支持针对URL域名的捕获和构造,这有些破坏了ASP.NET Routing所制定的“协议”(ASP.NET Routing只支持Path),因此在上一篇文章中,我们需要自己构造一个辅助方法来获得一个“包含域名”的URL。不过根据尽可能强类型的原则,我们应该使用的是类似于MvcFutures中定义的基于表达式树的辅助方法。由于MvcFutures已经提供了非常充足的辅助功能,因此这其实并不需要耗费我们多少代价。

上一次我们编写了这样的辅助方法:

public static string ActionEx(this UrlHelper helper, string action, object routeValues)
{
    var values = routeValues == null ?
        new RouteValueDictionary() : 
        new RouteValueDictionary(routeValues);
    values.Add("action", action);
    values.Add("controller", helper.RequestContext.RouteData.Values["controller"]);

    return helper.GetRouteUrl(values);
}

private static string GetRouteUrl(this UrlHelper helper, RouteValueDictionary values)
{
    var pathData = helper.RouteCollection.GetPath(helper.RequestContext, values);
    var url = pathData.VirtualPath;
    return IsAbsolute(url) ? url : "/" + url;
}

但是根据我们的需要,我们应该设法编写如下的代码:

<a href="<%= Url.ActionEx<HomeController>(c => c.Index()) %>">Home</a>

那么,这个ActionEx方法的签名应该是什么样的呢?从一个方法的调用方式上得出它的签名也是构造良好API的必要能力。在这里,我们可以把ActionEx方法的签名定成:

public static string ActionEx<TController>(
    this UrlHelper helper, Expression<Action<TController>> action)
    where TController : Controller
{
    return ActionEx(helper, action, null);
}

public static string ActionEx<TController>(
    this UrlHelper helper, Expression<Action<TController>> action, object routeValues)
    where TController : Controller
{
    ...
}

与原来的ActionEx方法不同,原来的ActionEx方法仅仅携带了一个字符串,而现在的action是一个表达式树,其中包含了大量的信息:调用哪个Controller中的哪个Action方法,并使用了哪些参数。例如,以下两种用法,最终生成的URL是相同的:

<%= Url.ActionEx("List", new { controller = "Post", id = 5, area = "blogs" }) %>
<%= Url.ActionEx<PostController>(c => c.List(5), new { area = "blogs" }) %>

这样,您应该就可以看出两种情况下,各种必要的数据是如何传递进来的。因此,新增的ActionEx方法应该是这样:

public static string ActionEx<TController>(
    this UrlHelper helper, Expression<Action<TController>> action, object routeValues)
    where TController : Controller
{
    var values = GetRouteValuesFromExpression(action);
    if (routeValues != null)
    {
        values.CopyFrom(new RouteValueDictionary(routeValues));
    }

    return helper.GetRouteUrl(values);
}

private static RouteValueDictionary GetRouteValuesFromExpression<TController>(
    Expression<Action<TController>> action) where TController : Controller
{
    ...
}

您应该可以料想得到,这里的关键是如何从表达式树中提取数据(即GetRouteValuesFromExpression方法的实现)。如果您不了解表达树,那么这方面可能略有难度。幸运的是,其实MvcFutures项目已经帮我们自带了充足的辅助功能:

private static RouteValueDictionary GetRouteValuesFromExpression<TController>(
    Expression<Action<TController>> action) where TController : Controller
{
    return ExpressionHelper.GetRouteValuesFromExpression(action);
}

就这样,结束了。当然,原有MvcFutures中缺少或不足的功能也直接带入了我们的项目中,下次我们便要改进这些功能。