我正在尝试实现一个自定义路由约束,它将检查数据库中的信息。我已经获得了路由约束,但是当我尝试将DbContext注入到类中以使用数据库时,我遇到了无参数构造函数问题。这是我的代码:
我的自定义路线约束:
public class DynamicPageRouteConstraint : IRouteConstraint
{
public const string ROUTE_LABEL = "dbDynamicRoute";
private ApplicationDbContext db;
public DynamicPageRouteConstraint(ApplicationDbContext _db)
{
db = _db;
}
public bool Match(HttpContext httpContext, IRouter route, string routeKey, RouteValueDictionary values, RouteDirection routeDirection)
{
List<string> items = db.DynamicPages.Select(p => p.Url).ToList();
if (items.Contains(values["dynamicPageFriendlyURL"]))
{
return true;
}
return false;
}
}
Startup.cs:
public void ConfigureServices(IServiceCollection services)
{
...
services.AddDbContext<ApplicationDbContext>(options => options.UseSqlServer(ConnectionString));
services.Configure<RouteOptions>(options =>
{
options.ConstraintMap.Add(DynamicPageRouteConstraint.ROUTE_LABEL, typeof(DynamicPageRouteConstraint));
});
services.AddMvc();
}
public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
...
app.UseMvc(routes =>
{
routes.MapRoute(
name: "DynamicPageRoutes",
template: "{*dynamicPageFriendlyURL:" + DynamicPageRouteConstraint.ROUTE_LABEL + "}",
defaults: new { controller = "DynamicPage", action = "Loader" },
constraints: new { dynamicPageFriendlyURL = new DynamicPageRouteConstraint(app.ApplicationServices.GetRequiredService<ApplicationDbContext>()) });
routes.MapRoute(
name: "default",
template: "{controller=Home}/{action=Index}/{id?}");
});
}
我的问题是:
现在的代码立即抱怨在startup.cs中遇到我的路由时错过了无参数构造函数。
如果我在RouteConstraint实现中放置一个无参数构造函数,我可以让代码工作,但是当命中路径时,它会被命中两次,一次注入我的依赖项,然后再次使用无参数构造函数,这意味着没有db对象实例化。
我不想放弃通过实例化一个新的dbcontext对象来注入dbContext,因为我需要指定一个连接字符串,这会在我们实现的dev / production连接字符串设置中引发一个大扳手。
我的问题:
为什么需要无参数构造函数?
为什么我的路线会被叫两次? (一次注入ctr,一次使用无参数ctr)
有没有更好的方法将依赖注入到IRouteConstraint的实现中?
有没有更好的方法可以做到这一点?我正在尝试为存储在数据库中的页面提供自定义URL。
任何帮助表示赞赏!谢谢!
答案 0 :(得分:3)
我认为你在MapRoute中应用了两次约束,一次在模板参数中(系统将在没有构造函数的情况下创建约束类的实例),然后在constraints参数中。我认为您的答案是将模板更改为
template: "{*dynamicPageFriendlyUrl}"
然后系统不会尝试在模板级别应用约束。
答案 1 :(得分:1)
在匹配功能
获取数据库服务var db = (ApplicationDbContext)httpContext.RequestServices.GetService(typeof(ApplicationDbContext));