在StackOverflow Podcast #54中,Jeff提到他们通过处理路由的方法上方的属性在StackOverflow代码库中注册其URL路由。听起来像一个很好的概念(菲尔哈克提出的关于路线优先权的警告)。
有人可以提供一些样本来实现这一目标吗?
此外,使用这种路由方式的任何“最佳做法”?
答案 0 :(得分:62)
更新 :这已发布在codeplex上。完整的源代码以及预编译的程序集可供下载。我还没有时间在网站上发布文档,所以这篇SO帖子现在已经足够了。
UPDATE :我添加了一些新属性来处理1)路由排序,2)路由参数约束,以及3)路由参数默认值。以下文字反映了此次更新。
我实际上已经为我的MVC项目做了类似的事情(我不知道Jeff是如何使用stackoverflow进行的)。我定义了一组自定义属性:UrlRoute,UrlRouteParameterConstraint,UrlRouteParameterDefault。它们可以附加到MVC控制器操作方法,以使路由,约束和默认值自动绑定到它们。
使用示例:
(注意这个例子有点人为,但它演示了这个特征)
public class UsersController : Controller
{
// Simple path.
// Note you can have multiple UrlRoute attributes affixed to same method.
[UrlRoute(Path = "users")]
public ActionResult Index()
{
return View();
}
// Path with parameter plus constraint on parameter.
// You can have multiple constraints.
[UrlRoute(Path = "users/{userId}")]
[UrlRouteParameterConstraint(Name = "userId", Regex = @"\d+")]
public ActionResult UserProfile(int userId)
{
// ...code omitted
return View();
}
// Path with Order specified, to ensure it is added before the previous
// route. Without this, the "users/admin" URL may match the previous
// route before this route is even evaluated.
[UrlRoute(Path = "users/admin", Order = -10)]
public ActionResult AdminProfile()
{
// ...code omitted
return View();
}
// Path with multiple parameters and default value for the last
// parameter if its not specified.
[UrlRoute(Path = "users/{userId}/posts/{dateRange}")]
[UrlRouteParameterConstraint(Name = "userId", Regex = @"\d+")]
[UrlRouteParameterDefault(Name = "dateRange", Value = "all")]
public ActionResult UserPostsByTag(int userId, string dateRange)
{
// ...code omitted
return View();
}
UrlRouteAttribute的定义:
/// <summary>
/// Assigns a URL route to an MVC Controller class method.
/// </summary>
[AttributeUsage(AttributeTargets.Method, Inherited = true, AllowMultiple = true)]
public class UrlRouteAttribute : Attribute
{
/// <summary>
/// Optional name of the route. If not specified, the route name will
/// be set to [controller name].[action name].
/// </summary>
public string Name { get; set; }
/// <summary>
/// Path of the URL route. This is relative to the root of the web site.
/// Do not append a "/" prefix. Specify empty string for the root page.
/// </summary>
public string Path { get; set; }
/// <summary>
/// Optional order in which to add the route (default is 0). Routes
/// with lower order values will be added before those with higher.
/// Routes that have the same order value will be added in undefined
/// order with respect to each other.
/// </summary>
public int Order { get; set; }
}
UrlRouteParameterConstraintAttribute的定义:
/// <summary>
/// Assigns a constraint to a route parameter in a UrlRouteAttribute.
/// </summary>
[AttributeUsage(AttributeTargets.Method, Inherited = true, AllowMultiple = true)]
public class UrlRouteParameterConstraintAttribute : Attribute
{
/// <summary>
/// Name of the route parameter on which to apply the constraint.
/// </summary>
public string Name { get; set; }
/// <summary>
/// Regular expression constraint to test on the route parameter value
/// in the URL.
/// </summary>
public string Regex { get; set; }
}
UrlRouteParameterDefaultAttribute的定义:
/// <summary>
/// Assigns a default value to a route parameter in a UrlRouteAttribute
/// if not specified in the URL.
/// </summary>
[AttributeUsage(AttributeTargets.Method, Inherited = true, AllowMultiple = true)]
public class UrlRouteParameterDefaultAttribute : Attribute
{
/// <summary>
/// Name of the route parameter for which to supply the default value.
/// </summary>
public string Name { get; set; }
/// <summary>
/// Default value to set on the route parameter if not specified in the URL.
/// </summary>
public object Value { get; set; }
}
对Global.asax.cs的更改:
只需调用RouteUtility.RegisterUrlRoutesFromAttributes函数,即可将对MapRoute的调用替换为:
public static void RegisterRoutes(RouteCollection routes)
{
routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
RouteUtility.RegisterUrlRoutesFromAttributes(routes);
}
RouteUtility.RegisterUrlRoutesFromAttributes的定义:
codeplex上有完整的来源。如果您有任何反馈或错误报告,请访问该网站。
答案 1 :(得分:44)
您还可以尝试使用AttributeRouting或github提供的nuget。
这是一个无耻的插件,因为我是项目作者。但是,如果我不是很高兴使用它。你可能也是。 github存储库wiki中有大量文档和示例代码。
使用这个库,你可以做很多事情:
我确信还有其他一些我忘了的东西。看看这个。通过nuget安装很容易。
注意:截至4月16日,AttributeRouting还支持新的Web API基础架构。以防您正在寻找可以处理的内容。 Thanks subkamran!
答案 2 :(得分:9)
<强> 1。下载RiaLibrary.Web.dll并在ASP.NET MVC网站项目中引用它
<强> 2。使用[Url]属性解压缩控制器方法:
public SiteController : Controller
{
[Url("")]
public ActionResult Home()
{
return View();
}
[Url("about")]
public ActionResult AboutUs()
{
return View();
}
[Url("store/{?category}")]
public ActionResult Products(string category = null)
{
return View();
}
}
顺便说一下,'?'登录'{?category}'参数意味着它是可选的。您不需要在路由默认值中明确指定它,这等于:
routes.MapRoute("Store", "store/{category}",
new { controller = "Store", action = "Home", category = UrlParameter.Optional });
第3。更新Global.asax.cs文件
public class MvcApplication : System.Web.HttpApplication
{
public static void RegisterRoutes(RouteCollection routes)
{
routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
routes.MapRoutes(); // This does the trick
}
protected void Application_Start()
{
RegisterRoutes(RouteTable.Routes);
}
}
如何设置默认值和约束?例如:强>
public SiteController : Controller
{
[Url("admin/articles/edit/{id}", Constraints = @"id=\d+")]
public ActionResult ArticlesEdit(int id)
{
return View();
}
[Url("articles/{category}/{date}_{title}", Constraints =
"date=(19|20)\d\d-(0[1-9]|1[012])-(0[1-9]|[12][0-9]|3[01])")]
public ActionResult Article(string category, DateTime date, string title)
{
return View();
}
}
如何订购?例如:强>
[Url("forums/{?category}", Order = 2)]
public ActionResult Threads(string category)
{
return View();
}
[Url("forums/new", Order = 1)]
public ActionResult NewThread()
{
return View();
}
答案 3 :(得分:3)
这篇文章只是为了扩展DSO的答案。
在将路由转换为属性时,我需要处理ActionName属性。所以在GetRouteParamsFromAttribute中:
ActionNameAttribute anAttr = methodInfo.GetCustomAttributes(typeof(ActionNameAttribute), false)
.Cast<ActionNameAttribute>()
.SingleOrDefault();
// Add to list of routes.
routeParams.Add(new MapRouteParams()
{
RouteName = routeAttrib.Name,
Path = routeAttrib.Path,
ControllerName = controllerName,
ActionName = (anAttr != null ? anAttr.Name : methodInfo.Name),
Order = routeAttrib.Order,
Constraints = GetConstraints(methodInfo),
Defaults = GetDefaults(methodInfo),
});
我也发现路线的命名不合适。该名称是使用controllerName.RouteName动态构建的。但我的路由名称是控制器类中的const字符串,我使用这些const也调用Url.RouteUrl。这就是为什么我真的需要属性中的路由名称作为路由的实际名称。
我要做的另一件事是将default和constraint属性转换为AttributeTargets.Parameter,以便我可以将它们粘贴到params。
答案 4 :(得分:0)
我已经将这两种方法合并为一个弗兰肯斯坦版本,适合任何想要它的人。 (我喜欢可选的param表示法,但也认为它们应该是默认/约束的单独属性,而不是全部混合成一个)。
http://github.com/djMax/AlienForce/tree/master/Utilities/Web/
答案 5 :(得分:0)
我需要使用AsyncController让ITCloud路由在asp.net mvc 2中工作 - 为此,只需在源代码中编辑RouteUtility.cs类并重新编译。你必须从第98行的行动名称中删除“已完成”
// Add to list of routes.
routeParams.Add(new MapRouteParams()
{
RouteName = String.IsNullOrEmpty(routeAttrib.Name) ? null : routeAttrib.Name,
Path = routeAttrib.Path,
ControllerName = controllerName,
ActionName = methodInfo.Name.Replace("Completed", ""),
Order = routeAttrib.Order,
Constraints = GetConstraints(methodInfo),
Defaults = GetDefaults(methodInfo),
ControllerNamespace = controllerClass.Namespace,
});
然后,在AsyncController中,使用熟悉的UrlRoute
和UrlRouteParameterDefault
属性修饰XXXXCompleted ActionResult:
[UrlRoute(Path = "ActionName/{title}")]
[UrlRouteParameterDefault(Name = "title", Value = "latest-post")]
public ActionResult ActionNameCompleted(string title)
{
...
}
希望能帮助有同样问题的人。