8

我将MediatR与以下类一起使用:

public class GetPostsRequest : IRequest<Envelope<GetPostsResponse>> {
  public Int32 Age { get; set; }
}

public class GetPostResponse {
  public String Title { get; set; }
  public String Content { get; set; }      
}

其中 Envelope 是一个包装类:

public class Envelope<T> {
  public List<T> Result { get; private set; } = new List<T>();  
  public List<Error> Errors { get; private set; } = new List<Error>();
}

由中介发送的 GetPostsRequest 由 Handler 执行:

public class GetPostsRequestHandler : IRequestHandler<GetPostsRequest, Envelope<GetPostsResponse>> {

  public async Task<Envelope<GetPostsResponse>> Handle(GetPostsRequest request, CancellationToken cancellationToken) {
  }

}

MediatR 允许使用在特定请求的处理程序之前执行的行为。我创建了一个ValidationBehavior如下:

public class ValidationBehavior<TRequest, TResponse> : IPipelineBehavior<TRequest, Envelope<TResponse>> 
    where TRequest : IRequest<Envelope<TResponse>> {

  private readonly IEnumerable<IValidator<TRequest>> _validators;

  public ValidationBehavior(IEnumerable<IValidator<TRequest>> validators) {
    _validators = validators;
  }

  public Task<Envelope<TResponse>> Handle(TRequest request, CancellationToken cancellationToken, RequestHandlerDelegate<Envelope<TResponse>> next) {

    ValidationContext context = new ValidationContext(request);

    List<Error> errors = _validators
      .Select(x => x.Validate(context))
      .SelectMany(x => x.Errors)
      .Select(x => new Error(ErrorCode.DataNotValid, x.ErrorMessage, x.PropertyName))
      .ToList();

    if (errors.Any())
      return Task.FromResult<Envelope<TResponse>>(new Envelope<TResponse>(errors));  

    return next();

  }

}

我在 ASP.NET Core 应用程序上注册了 ValidationBehavior:

services.AddScoped(typeof(IPipelineBehavior<,>), typeof(ValidationBehavior<,>));

当我调用 API 时,出现以下错误:

An unhandled exception has occurred while executing the request.
System.ArgumentException: GenericArguments[0], 'TRequest', on 'ValidationBehavior`2[TRequest,TResponse]' violates the constraint of type 'TRequest'. 
---> System.TypeLoadException: GenericArguments[0], 'TRequest', on 'ValidationBehavior`2[TRequest,TResponse]' violates the constraint of type parameter 'TRequest'.

我错过了什么?

4

2 回答 2

2
services.AddScoped(typeof(IPipelineBehavior<,>), typeof(ValidationBehavior<,>));

我不认为这条线做你期望它做的事。

在您的示例中,您的请求类型是 a IRequest<Envelope<Response>>,因此当 MediatR 查找管道行为时,它会查找IPipelineBehavior<Request, Envelope<Response>>.

由于 DI 容器中有一个开放通用注册,IPipelineBehavior<,>因此将使用该实现。因此,泛型类型参数 fromIPipelineBehavior<,>将用于实例化ValidationBehavior<,>.

所以对于IPipelineBehavior<Request, Envelope<Response>>,这将创建一个ValidationBehavior<Request, Envelope<Response>>. 这意味着在您的通用实现中,TRequestwill beRequestTResponsewill be Envelope<Response>。然而,这意味着您的类型上的类型约束意味着TRequest,或者Request在这种情况下,需要实现IRequest<Envelope<Envelope<Response>>>. 当然,事实并非如此,因此可以解释您所看到的类型约束违规。

不幸的是,这意味着您无法按照您希望的方式进行操作。您不能对泛型类型施加类型约束,至少不能对Envelope<T>.

于 2018-08-29T15:25:25.157 回答
0

ValidationBehavior指定TRequest必须是IRequest<Envelope<TResponse>>
您的请求实施IRequest<Envelope<Response>> (不匹配 -Envelope<TResponse>不是Envelope<Response>

但是,您的 ValidationBehavior 实现IPipelineBehavior<TRequest, Envelope<TResponse>> 并具有约束where TRequest : IRequest<Envelope<TResponse>>

所以,Envelope<TResponse>改为Envelope<Response>

于 2018-08-29T15:13:44.470 回答