在Spring的Controller中,我们通过@RequestParam
或@RequestBody
就可以将请求中的参数映射到控制层具体的参数中,那么这个是怎么实现的呢?如果我现在控制层中的某个参数的值是从Redis
中来,那么应该如何实现呢?
从上图中可以我们的参数最终会通过HandlerMethodArgumentResolver
来解析,那么知道了这个后,我们就可以实现自己的参数解析了。
如果我们控制层方法的参数中存在@Redis
标注,那么此参数的值应该从redis中获取,不用从请求参数中获取。
从上图中可知@Redis(key = "redisKey") String redisValue
这个参数就需要从Redis
中获取。
此处我们不会真的从
Redis
中获取值,模拟从Redis中获取值即可。
1、编写一个Redis注解
@Target({ElementType.PARAMETER}) @Retention(RetentionPolicy.RUNTIME) public @interface Redis { /** * redis中的Key */ String key(); }
在控制层的方法上,被此处注解标注的方法参数,都从Redis中获取,都走我们自己定义的参数解析器。
2、编写参数解析类
package com.huan.study.argument.resolver; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.core.MethodParameter; import org.springframework.web.bind.support.WebDataBinderFactory; import org.springframework.web.context.request.NativeWebRequest; import org.springframework.web.method.support.HandlerMethodArgumentResolver; import org.springframework.web.method.support.ModelAndViewContainer; import javax.servlet.http.HttpServletRequest; import java.util.Random; /** * 从redis中获取值放入到参数中 * */ public class RedisMethodArgumentResolver implements HandlerMethodArgumentResolver { private static final Logger log = LoggerFactory.getLogger(RedisMethodArgumentResolver.class); /** * 处理参数上存在@Redis注解的 */ @Override public boolean supportsParameter(MethodParameter parameter) { return parameter.hasParameterAnnotation(Redis.class); } /** * 解析参数 */ @Override public Object resolveArgument(MethodParameter parameter, ModelAndViewContainer mavContainer, NativeWebRequest webRequest, WebDataBinderFactory binderFactory) throws Exception { // 获取http request HttpServletRequest request = webRequest.getNativeRequest(HttpServletRequest.class); log.info("当前请求的路径:[{}]", request.getRequestURI()); // 获取到这个注解 Redis redis = parameter.getParameterAnnotation(Redis.class); // 获取在redis中的key String redisKey = redis.key(); // 模拟从redis中获取值 String redisValue = "从redis中获取的值:" + new Random().nextInt(100); log.info("从redis中获取到的值为:[{}]", redisValue); // 返回值 return redisValue; } }
1、通过supportsParameter
方法判断我们应该处理哪些参数,此处处理的是参数上存在@Redis
注解的。
2、通过resolveArgument
方法,获取到参数的具体的值。比如从Redis中获取,代码中没有真的从Redis中获取,只是模拟从Redis中获取。
3、配置到Spring的上下文中
此处我们最好将我们自己的参数解析器放置在第一位,否则可能会有问题。下方提供了2种方式,第一种方式无法达到我们的需求、我们采用第二种方式来实现
1、通过WebMvcConfigurer实现
@Configuration public class WebMvcConfig implements WebMvcConfigurer { /** * 这个地方加载的顺序是在默认的HandlerMethodArgumentResolver之后的 */ @Override public void addArgumentResolvers(List<HandlerMethodArgumentResolver> resolvers) { resolvers.add(new RedisMethodArgumentResolver()); } }
从上图可知,我们自己的参数解析器不是在第一位,这样可能达不到我们想要的效果,此处不考虑这种方式。
2、通过BeanPostProcessor来实现
BeanPostProcessor
可以在一个Bean完全初始化之后来执行一些操作,此处我们通过这种方式,将我们自己的参数解析器放置在第一位。
@Component static class CustomHandlerMethodArgumentResolverConfig implements BeanPostProcessor { @Override public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { if (bean instanceof RequestMappingHandlerAdapter) { final RequestMappingHandlerAdapter adapter = (RequestMappingHandlerAdapter) bean; final List<HandlerMethodArgumentResolver> argumentResolvers = Optional.ofNullable(adapter.getArgumentResolvers()) .orElseGet(ArrayList::new); final ArrayList<HandlerMethodArgumentResolver> handlerMethodArgumentResolvers = new ArrayList<>(argumentResolvers); // 将我们自己的参数解析器放置到第一位 handlerMethodArgumentResolvers.add(0, new RedisMethodArgumentResolver()); adapter.setArgumentResolvers(Collections.unmodifiableList(handlerMethodArgumentResolvers)); return adapter; } return bean; } }
从上图可知,我们自己的参数解析器在第一位,这样就达到我们想要的效果,此处使用这种方式。
4、编写一个简单的控制层
/** * @author huan.fu 2021/12/7 - 下午3:36 */ @RestController public class RedisArgumentController { private static final Logger log = LoggerFactory.getLogger(RedisArgumentController.class); @GetMapping("redisArgumentResolver") public void redisArgumentResolver(@RequestParam("hello") String hello, @Redis(key = "redisKey") String redisValue) { log.info("控制层获取到的参数值: hello:[{}],redisValue:[{}]", hello, redisValue); } }
该控制层比较简单,对外提供来一个简单的apihttp://localhost:8080/redisArgumentResolver?hello=123
。该api存在2个参数hello
和redisValue
,其中hello
参数的值是从请求参数中获取,redisValue
的值是从我们自己定义的参数
解析器中获取。
curl http://localhost:8080/redisArgumentResolver?hello=123
由上图可知我们自己定义的参数解析器工作了。
热门文章
- 我想看动物园图片(我想看动物园的图片)
- 3月31日18.8M/S|Shadowrocket/V2ray/SSR/Clash免费节点每天更新订阅链接,翻墙机场推荐分享
- 猫三联狂犬疫苗猫吐绿(猫咪打完狂犬疫苗吐黄水和白沫)
- 动物医院电话号码多少啊怎么查询(动物医院的电话号码是多少)
- spring boot web项目
- MySQL CONCAT_WS()函数的用途是什么?
- 3月3日22.8M/S|Shadowrocket/V2ray/Clash/SSR免费节点每天更新订阅链接,翻墙机场推荐分享
- 《Java—Socket二进制通讯读取一行》
- 3月17日22.2M/S|V2ray/Shadowrocket/SSR/Clash免费节点每天更新订阅链接,翻墙机场推荐分享
- 中国兽药网官方网站 中国兽药网官方网站查询