(WebFlux)001、如何自定义注解实现功能

如何实现自定义注解

一、背景

最近在项目又在压测,但基于Http请求类型的校验过多,已有想法把Http请求换成Spring中的WebClient,但是由于不是原配(SpringWebFlux + WebClient),如果采用WebClent....block()这样的实现方式,阻塞获取结果,老是觉得别扭,所以就想把SpringMVC换成SpringWebFlux(新手上路),大胆尝试,直接发车。

现在把换的过程中的问题和解决方式列出来,供大家参考。

SpringBoot版本号: 2.6.10

二、正文

在使用SpringMVC实现业务逻辑时,我们经常会采用一些自定义注解,通过自定义注解实现访问URL过滤、鉴权等一些列功能。但是在WebFlux中如何实现呢?

2.1 MVC 实现方案

在MVC中,我们一般直接实现HandlerInterceptor,然后实现HandlerInterceptor#preHandle()方法即可,然后在其中实现自己的逻辑。代码如下、

先实现自定义注解:

/**
 * 

校验权限

* * @author [email protected] * @since 2022/7/24 */ @Target({ElementType.TYPE, ElementType.METHOD}) // 可用于方法和类上 @Retention(RetentionPolicy.RUNTIME) @Documented public @interface CheckPermission { boolean check() default true; }

为什么要标记在方法和类上呢? 那是因为有可能某一个类的方法都需要CheckPermission 或者都不check,直接标记在类上就省事了。

接下来是实现拦截器:

/**
 * 

拦截器

* * @author [email protected] * @since 2022/7/24 */ @Component public class CheckPermissionInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { if (handler instanceof HandlerMethod) { HandlerMethod methodHandle = (HandlerMethod) handler; CheckPermission permission = methodHandle.getMethodAnnotation(CheckPermission.class); if (Objects.isNull(permission)) { // handler 所有bean 上面寻找 permission = AnnotationUtils.findAnnotation(methodHandle.getBeanType(), CheckPermission.class); } if (Objects.nonNull(permission)) { // TODO do something.. } } return true; } }

当我们在当前Handler上没有找到注解的的时候,我们就去所在类上寻找,判断是否需要校验。偷鸡,减少代码注解标记量(心理美滋滋)。

2.2 WebFlux 实现方案

MVC的实现方式对我们来说可谓是手到擒来,闭着眼睛也能把代码敲完(吹嘘居多...),这么常见、基础、重要的功能,在WebFlux中是如何实现的呢?

我在翻了好多文档后,终于找到了答案,当然要分享给大家了哇,原文采用的kotlin,我就用JAVA了。原文出处

在WebFlux中没有HandlerInterceptor,我们需要采用提供的WebFilter实现功能。注解同上,实现的逻辑代码如下。

/**
 * 

WebFlux实现方案

* * @author [email protected] * @since 2022/7/24 */ @Component public class CheckPermissionWebFilter implements WebFilter { @Autowired // 关键,通过RequestMappingHandlerMapping 我们可以获取到MethodHandler private RequestMappingHandlerMapping handlerMapping; @Override public Mono filter(ServerWebExchange exchange, WebFilterChain chain) { return handlerMapping.getHandler(exchange).switchIfEmpty(chain.filter(exchange)) .flatMap(handler -> { // 熟悉的味道,将handler转换成HandlerMethod if (handler instanceof HandlerMethod) { HandlerMethod methodHandle = (HandlerMethod) handler; CheckPermission permission = methodHandle.getMethodAnnotation(CheckPermission.class); if (Objects.isNull(permission)) { // handler 所有bean 上面寻找 permission = AnnotationUtils.findAnnotation(methodHandle.getBeanType(), CheckPermission.class); } if (Objects.nonNull(permission)) { // TODO do something.. } } return chain.filter(exchange); } ); } }

可以看到,在实现过程中,最关键的是要知道有org.springframework.web.reactive.result.method.annotation.RequestMappingHandlerMapping这个类,知道之后,我们就可以通过其获取到Handler。拿到Handler后就可以按照我们自己的套路来实现具体的业务逻辑了。

2.3 对比

通过对比MVC和WebFlux的实现方式来看,Spring对两种都做了比较好的支持,都是通过获取到HandlerMethod,然后在对具体的逻辑处理。对比后发现,代码相似度达到80%~90%。这也要求我们要做的可能是更多的知道其有的API,这样可以减少我们在对项目做转换时带来的时间成本问题。至于实现,我们可以在熟悉使用时,一步步探索,挖掘实现逻辑,提高知识存储量。

三、总结

尝试新东西的时候是一个不断探索,学习的过程。多看,多搜,多学。

你可能感兴趣的:((WebFlux)001、如何自定义注解实现功能)