我需要为特定类Foo
定制模型绑定,该类涉及通过一些额外的后处理扩展正常绑定逻辑(例如,有条件地将空集合字段设置为空集合)。我想将此逻辑添加到模型绑定中,以便结果可用于操作过滤器等。
最直接的方法是从ComplexTypeModelBinder
派生并覆盖BindModelAsync
。但是,遗憾的是,该方法不是虚拟的。
组成是下一个选择。我试图创建一个拥有或获得FooModelBinder
实例的ComplexTypeModelBinder
类。但是,我无法弄清楚如何注入或解析ComplexTypeModelBinder
。这可能吗?有没有更好的方法来扩展ComplexTypeModelBinder
的功能?
答案 0 :(得分:4)
我终于意识到模型绑定器是从模型绑定器提供程序获得的,而不是通过依赖注入。要正确实例化我的FooModelBinder
,我需要创建一个FooModelBinderProvider
。要正确获取ComplexTypeModelBinder
的组合实例,我的提供商需要访问ComplexTypeModelBinderProvider
。换句话说,要构建模型绑定器,还需要组合模型绑定器提供程序。
这是提供者。请注意,我们不需要指定注入提供程序的确切类型,因为我们只是简单地包装另一个模型绑定程序的现有功能。
public class FooModelBinderProvider : IModelBinderProvider
{
private readonly IModelBinderProvider workerProvider;
public FooModelBinderProvider(IModelBinderProvider workerProvider)
{
this.workerProvider = workerProvider;
}
public IModelBinder GetBinder(ModelBinderProviderContext context)
{
if (context == null)
{
throw new ArgumentNullException(nameof(context));
}
if (context.Metadata.ModelType == typeof(Foo))
{
return new FooModelBinder(this.workerProvider.GetBinder(context));
}
return null;
}
}
这是活页夹。请注意,我们在BindModelAsync
中做的第一件事是蹦床进入“工人”活页夹。
public class FooModelBinder : IModelBinder
{
private readonly IModelBinder worker;
public FooModelBinder(IModelBinder worker)
{
this.worker = worker;
}
public async Task BindModelAsync(ModelBindingContext bindingContext)
{
await this.worker.BindModelAsync(bindingContext);
if (!bindingContext.Result.IsModelSet)
{
return;
}
var foo = bindingContext.Result.Model as Foo;
if (foo == null)
{
throw new InvalidOperationException($"Expected {bindingContext.ModelName} to have been bound by ComplexTypeModelBinder");
}
// NOW DO SOME INTERESTING POST-PROCESSING
}
}
最后,这是注册自定义活页夹的方法:
services.AddMvc(options =>
{
var workerProvider = options.ModelBinderProviders.First(p => p.GetType() == typeof(ComplexTypeModelBinderProvider));
options.ModelBinderProviders.Insert(options.ModelBinderProviders.IndexOf(workerProvider), new FooModelBinderProvider(workerProvider));
})