gpt4 book ai didi

c# - 路由代码中指定的 Controller /方法有 "specifications"是不是很糟糕?

转载 作者:行者123 更新时间:2023-11-30 22:20:07 24 4
gpt4 key购买 nike

我正在为 ASP.Net 设计一个替代的 MVC 框架。我对该框架的部分目标是尽可能少的“魔法”。我唯一的想法是将表单、查询字符串等内容绑定(bind)到一个普通的 ol' 类(带有一些用于忽略、转换等的可选属性)。因此,我绝对不进行类/方法检测。一切都必须非常明确。我经历了大约 3 次 API“形状”迭代。前两个实现了我没有魔法的目标,但它非常冗长且不易阅读……而且 Controller 通常不得不做 MVC 框架应该做的繁重工作。

所以,现在在这第三次迭代中,我正在非常努力地把它做好。有一点有争议我做的不同是路由代码。因为一切都是显式的并且不鼓励反射,所以我无法在 Controller 中搜索某些属性来解析路由。一切都必须在路由级别指定。在第一次迭代中,这并没有完成,但它导致了极其繁琐和冗长的 Controller ......

现在,我拥有用于​​指定路线的流畅 API。它已经超出了我最初的想象,但现在可以作为一种方式来指定 Controller 的方法能够做什么以及它应该接受什么。

进入实际代码。实现是无关紧要的。您真正需要知道的唯一一件事是涉及很多泛型类型。因此,这是一些路由的快速示例:

var router=new Router(...);
var blog=router.Controller(() => new BlogController());
blog.Handles("/blog/index").With((ctrl) => ctrl.Index());
blog.Handles("/blog/{id}").With((ctrl,model) => ctrl.View(model["id"])).WhereRouteLike((r) => r["id"].IsInteger()); //model defaults to creating a dictionary from route parameters
blog.Handles("/blog/new").UsingFormModel(() => new BlogPost()).With((ctrl, model) => ctrl.NewPost(model)); //here model would be of type BlogPost. Also, could substitue UsingRouteModel, UsingQueryStringModel, etc

还有一些其他方法可以实现,例如 WhereModelIsLike 或一些对模型进行验证的方法。但是,这种“规范”属于路由层吗?路由层应该指定哪些限制?应该留给 Controller 验证什么?

我是不是让路由层担心太多了?

最佳答案

我认为路由太冗长了。我不想为 20 个 Controller 编写那种代码。特别是,因为它真的是重复的。
我在这里看到的问题是,即使是默认情况也需要冗长的声明。只有在特殊情况下才需要那些冗长的声明。
它具有表现力和可读性,但您可能需要考虑打包高级功能。

看看下面的规范。这只是针对单个 Controller 中的单个操作:

blog.Handles("/blog/new")
.UsingFormModel(() => new BlogPost())
.With((ctrl, model) => ctrl.NewPost(model))
.WhereModelIsLike(m => m.Status == PostStatus.New);

稍微减少代码量的一种方法是允许指定根文件夹:

var blog=router.Controller(() => new BlogController(), "/blog");
blog.Handles("index").Wi..
blog.Handles("{id}").Wit..
blog.Handles("new").Usin..

另一个减少默认情况代码的想法是为每个默认操作引入一个接口(interface)。 Controller 需要为支持的操作实现接口(interface):

可能是这样的:

public interface ISupportIndex
{
void Index();
}

public interface ISupportSingleItem
{
void View(int id);
}

现在,您可以提供类似 blog.HandlesIndex(); 的方法, blog.HandlesSingleItem(); .
这些方法返回与您现有方法相同的东西,因此可以进一步细化结果。
它们可以设计为仅在 Controller 实际实现接口(interface)时才可用的扩展方法。为此,router.Controller 的返回类型将需要与 Controller 作为通用参数的协变接口(interface),即像这样的东西:

IControllerRoute<out TController>

例如扩展方法HandlesIndex将像这样实现:

public static IRouteHandler HandlesIndex(
this IControllerRoute<ISupportIndex> route)
{
// note: This makes use of the "root" as suggested above:
// It only specifies "index", not "/someroot/index".
return route.Handles("index").With(x => x.Index);
}

致力于 IControllerRoute<ISupportIndex> , 仅在 Controller 实际支持的情况下显示。

博客 Controller 的路由可能如下所示:

blog.HandlesIndex();
blog.HandlesSingleItem();

// Uses short version for models with default constructor:
blog.HandlesNew<BlogPost>().UsingFormModel();
// The version for models without default constructor could look like this:
//blog.HandlesNew<BlogPost>().UsingFormModel(() => new BlogPost(myDependency));

添加验证规则也可以更简洁一些:

blog.HandlesNew<BlogPost>().UsingFormModel()
.When(m => m.Status == PostStatus.New);

如果规范更复杂,它可以打包在自己的实现IModelValidation的类中.现在使用该类:

blog.HandlesNew<BlogPost>().UsingFormModel()
.WithValidator<NewBlogPostValidation>();

我的所有建议都只是让您当前的方法更容易处理的方法,所以我想到目前为止,它并没有真正回答您的实际问题。我现在这样做:

我喜欢我的 Controller 尽可能干净。将验证规则放在路由上对我来说看起来非常好,因为 Controller 操作现在可以假设它只使用有效数据调用。我会继续采用这种方法。

关于c# - 路由代码中指定的 Controller /方法有 "specifications"是不是很糟糕?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15152815/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com