如何扩展ComplexTypeModelBinder

时间:2017-12-04 22:28:13

标签: c# asp.net-core asp.net-core-mvc model-binding

我需要为特定类Foo定制模型绑定,该类涉及通过一些额外的后处理扩展正常绑定逻辑(例如,有条件地将空集合字段设置为空集合)。我想将此逻辑添加到模型绑定中,以便结果可用于操作过滤器等​​。

最直接的方法是从ComplexTypeModelBinder派生并覆盖BindModelAsync。但是,遗憾的是,该方法不是虚拟的。

组成是下一个选择。我试图创建一个拥有或获得FooModelBinder实例的ComplexTypeModelBinder类。但是,我无法弄清楚如何注入或解析ComplexTypeModelBinder。这可能吗?有没有更好的方法来扩展ComplexTypeModelBinder的功能?

1 个答案:

答案 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));
  })