c# - ASP.NET Core 2.2 中的端点路由不起作用
问题描述
更新 (2)
@poke 似乎已经弄清楚了,这似乎是端点路由本身的一个问题,{*url}
如果其他更高的路由存在歧义,那么它有利于它。
更新 (1)
@poke 评论说我在控制器中缺少t{*url}
的路线上有错字。修复该路线后,该路线开始工作并且该操作正在工作。{*url}
DefaultController.Gone
但!现在奇怪的行为又开始出现了。{*url}
修复后,导航到/settings
应该与路线匹配的{controller}/{action}
路线失败并回退到{*url}
路线。
如果我{*url}
从注册中删除路线,然后/settings
再次工作。该{action}
路线继续不起作用。
原来的
请原谅问题的长度,但我试图提供尽可能多的信息。
我正在为自己开发一个 ASP.NET Core 2.2 博客应用程序,并且在路由正常工作时遇到了莫名其妙的问题。在对着屏幕大喊大叫半天之后,我决定退后一步,开始一个完全孤立的新项目。不知何故,问题在新项目中仍然存在。我已经把它剥离成一个饥饿的骨架,但我仍然无法让路线工作。我试图设置的路线是:
settings/{controller}/{id:int}/{action} - works
settings/{controller}/{action} - works
blog/{*slug} - works
blog/{skip:int?} - works
{controller}/{action} - works
{action} - doesn't work
{*url} - doesn't work
具体来说,我在最后两条路线上遇到了问题。
该{action}
路由不会为简单的操作生成,例如DefaultController.About
即使它没有约束,它所拥有的只是默认值,PostsController.List
因为我希望为根 URL 显示一个帖子列表。
{*url}
只是似乎根本不起作用。我想将它用作我的最终后备,它默认为DefaultController.Gone
,但如果我只是在键盘上敲击一些无意义的 URL,我得到的只是 404 错误。
我觉得问题在于DefaultController
theAbout
和Gone
动作都在其中,而且似乎都没有工作,但我似乎无法弄清楚如何。它实际上什么都不做,只是渲染视图,就像其他控制器一样。
下面是精简项目的代码。如果有人可以将它旋转并告诉我我在哪里失败,我将非常感激,因为我似乎无法弄清楚。
程序.cs
public sealed class Program {
public static async Task Main(
string[] args) => await WebHost.CreateDefaultBuilder(args)
.UseStartup<Startup>()
.Build()
.RunAsync();
}
启动.cs
public class Startup {
public void ConfigureServices(
IServiceCollection services) {
services.AddMvc()
.SetCompatibilityVersion(CompatibilityVersion.Latest);
}
public void Configure(
IApplicationBuilder app) {
app.UseMvc(
r => {
// /settings/{controller}/{id}/{action}
r.MapRoute("600", "settings/{controller}/{id:int}/{action}", null, new {
controller = "Categories|Tags"
});
// /settings/{controller}/{action}
r.MapRoute("500", "settings/{controller}/{action}", null, new {
controller = "Categories|Tags"
});
// /blog/*
r.MapRoute("400", "blog/{*slug}", new {
action = "Show",
controller = "Posts"
});
// /blog/{skip}
r.MapRoute("300", "blog/{skip:int?}", new {
action = "List",
controller = "Posts"
});
// /{controller}/{action}
r.MapRoute("200", "{controller}/{action=Default}", null, new {
controller = "Settings|Tools"
});
// /{action}
r.MapRoute("100", "{action}", new {
action = "List",
controller = "Posts"
});
// /*
r.MapRoute("-1", "{*url}", new {
action = "Gone",
conroller = "Default"
});
});
}
}
类别控制器.cs
public sealed class CategoriesController :
Controller {
[HttpGet]
public IActionResult Add() => Content("Category added");
[HttpGet]
public IActionResult Remove(
int id) => Content($"Category {id} removed");
}
DefaultController.cs
public sealed class DefaultController :
Controller {
[HttpGet]
public IActionResult About() => View();
[HttpGet]
public IActionResult Gone() => View();
}
About.cshtml(默认)
<h1>DEFAULT.ABOUT</h1>
Gone.cshtml(默认)
<h1>DEFAULT.GONE</h1>
PostsController.cs
public sealed class PostsController :
Controller {
[HttpGet]
public IActionResult List(
int? skip) => View();
[HttpGet]
public IActionResult Show(
string slug) => View();
}
List.cshtml(帖子)
<h1>POSTS.LIST</h1>
<a asp-action="Show" asp-controller="Posts" asp-route-slug="test-test-test">Show a Post</a>
Show.cshtml(帖子)
<h1>POSTS.SHOW</h1>
设置控制器.cs
public sealed class SettingsController :
Controller {
[HttpGet]
public IActionResult Default() => View();
}
Default.cshtml(设置)
<h1>SETTINGS.DEFAULT</h1>
<a asp-action="Add" asp-controller="Categories">Add a Category</a>
<br />
<a asp-action="Remove" asp-controller="Categories" asp-route-id="1">Remove a Category</a>
<hr />
<a asp-action="Add" asp-controller="Tags">Add a Tag</a>
<br />
<a asp-action="Remove" asp-controller="Tags" asp-route-id="1">Remove a Tag</a>
标签控制器.cs
public sealed class TagsController :
Controller {
[HttpGet]
public IActionResult Add() => Content("Tag added");
[HttpGet]
public IActionResult Remove(
int id) => Content($"Tag {id} removed");
}
工具控制器.cs
public sealed class ToolsController :
Controller {
[HttpGet]
public IActionResult Default() => View();
}
默认.cshtml
<h1>TOOLS.DEFAULT</h1>
_Layout.cshtml
<!DOCTYPE html>
<html>
<head>
<title>@ViewBag.Title</title>
</head>
<body>
<a asp-action="List" asp-controller="Posts">Blog</a>
<br />
<a asp-action="Default" asp-controller="Tools">Tools</a>
<br />
<a asp-action="About" asp-controller="Default">About</a>
<br />
<a asp-action="Default" asp-controller="Settings">Settings</a>
<br />
@RenderBody()
</body>
</html>
解决方案
{action}
- 不工作
这个不起作用,因为它必须与实际动作相匹配。因此,它适用于/Show
或/List
因为您在PostsController
. 它也适用于,/
因为action
默认为List
.
{*url}
- 不工作
如果您设置默认值controller
而不是conroller
:
r.MapRoute("-1", "{*url}", new
{
action = "Gone",
controller = "Default"
});
<a asp-action="About" asp-controller="Default">About</a>
请注意,此路由也将不匹配,因为没有指向该操作的路由。路线{controller}/{action}
被限制为SettingsController
and ToolsController
,因此路线将不匹配。您将需要调整约束或添加另一条路线才能使其正常工作。
顺便提一句。作为一般建议:您可能已经注意到,管理如此多的路由映射变得相当复杂。将属性路由与显式路由一起使用通常更容易。您还可以将它们与基于模板的路由混合使用,以获得两全其美的效果。
奇怪
/settings
的是,应该与{controller}/{action}
路线匹配的现在失败并退回到{*url}
路线。如果我{*url}
从注册中删除路线,然后/settings
再次工作。
settings/{controller}/{action}
这似乎是合并和{controller}/{action=Default}
路由的副作用。
我现在已经调试了一段时间,这似乎是端点路由的一个错误,它有利于捕获所有路由,尽管它稍后会被注册。
不幸的是,众所周知,ASP.NET Core 2.2 中的端点路由会在一些特殊情况下中断,这就是为什么要针对 3.0 对其进行修改,这有望解决所有问题。话虽如此,我已经打开了一个关于这个特定问题的问题并报告了我的发现。也许有一个简单的解决方案。
一种简单的解决方法是更改settings/{controller}/{action}
路由模板以使用除 之外的前缀settings
,这样就不再有歧义。这似乎可以解决问题。
推荐阅读
- c# - 嵌套类:在父类中调用子类属性
- image - 如何在 React Native 中预加载图像?
- zsh - 更改默认 Zsh 完成功能
- twilio - 是否可以仅将 Twilio 状态电子邮件过滤到美国,因此我不会收到有关我们目前未运营的国家/地区的状态电子邮件
- apache-poi - POI - Excel 单元格中图像的固定宽度
- amazon-web-services - 将现有 S3 存储桶与 Beanstalk 关联
- vue.js - Vue 3 - 通过 $router 将查询字符串参数添加到 URL
- php - 未检索或存储隐藏的表单值
- php - htaccess 重写规则删除子目录和 .php 扩展名
- android - 带有操作按钮和图像的 Flutter 本地通知