SpringBoot:基于redis自定义注解实现后端接口防重复提交校验

x33g5p2x  于2022-02-07 转载在 Spring  
字(16.9k)|赞(0)|评价(0)|浏览(516)

SpringBoot:基于redis自定义注解实现后端接口防重复提交校验

一、添加依赖

  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-redis</artifactId>
  4. <version>1.4.4.RELEASE</version>
  5. </dependency>
  6. <dependency>
  7. <groupId>redis.clients</groupId>
  8. <artifactId>jedis</artifactId>
  9. <version>3.6.3</version>
  10. </dependency>
  11. <dependency>
  12. <groupId>com.alibaba</groupId>
  13. <artifactId>fastjson</artifactId>
  14. <version>1.2.75</version>
  15. </dependency>

二、代码实现

1.构建可重复读取inputStream的request

  1. package com.hl.springbootrepetitionsubmit.servlet;
  2. import javax.servlet.*;
  3. import javax.servlet.http.*;
  4. import java.io.*;
  5. import java.nio.charset.StandardCharsets;
  6. /**
  7. * 构建可重复读取inputStream的request
  8. */
  9. public class RepeatedlyRequestWrapper extends HttpServletRequestWrapper {
  10. private final String body;
  11. public RepeatedlyRequestWrapper(HttpServletRequest request, ServletResponse response) throws IOException {
  12. super(request);
  13. request.setCharacterEncoding("UTF-8");
  14. response.setCharacterEncoding("UTF-8");
  15. body = getBodyString(request);
  16. }
  17. @Override
  18. public BufferedReader getReader() throws IOException {
  19. return new BufferedReader(new InputStreamReader(getInputStream()));
  20. }
  21. public String getBody() {
  22. return body;
  23. }
  24. @Override
  25. public ServletInputStream getInputStream() throws IOException {
  26. final ByteArrayInputStream bais = new ByteArrayInputStream(body.getBytes(StandardCharsets.UTF_8));
  27. return new ServletInputStream() {
  28. @Override
  29. public int read() throws IOException {
  30. return bais.read();
  31. }
  32. @Override
  33. public int available() throws IOException {
  34. return body.length();
  35. }
  36. @Override
  37. public boolean isFinished() {
  38. return false;
  39. }
  40. @Override
  41. public boolean isReady() {
  42. return false;
  43. }
  44. @Override
  45. public void setReadListener(ReadListener readListener) {
  46. }
  47. };
  48. }
  49. /**
  50. * 获取Request请求body内容
  51. *
  52. * @param request
  53. * @return
  54. */
  55. private String getBodyString(ServletRequest request) {
  56. StringBuilder sb = new StringBuilder();
  57. BufferedReader reader = null;
  58. try (InputStream inputStream = request.getInputStream()) {
  59. reader = new BufferedReader(new InputStreamReader(inputStream, StandardCharsets.UTF_8));
  60. String line = "";
  61. while ((line = reader.readLine()) != null) {
  62. sb.append(line);
  63. }
  64. } catch (IOException e) {
  65. e.printStackTrace();
  66. } finally {
  67. if (reader != null) {
  68. try {
  69. reader.close();
  70. } catch (IOException e) {
  71. e.printStackTrace();
  72. }
  73. }
  74. }
  75. return sb.toString();
  76. }
  77. }

2.重写request

  1. package com.hl.springbootrepetitionsubmit.filter;
  2. import com.hl.springbootrepetitionsubmit.servlet.RepeatedlyRequestWrapper;
  3. import org.springframework.http.MediaType;
  4. import org.springframework.util.StringUtils;
  5. import javax.servlet.*;
  6. import javax.servlet.http.HttpServletRequest;
  7. import java.io.IOException;
  8. /**
  9. * 过滤器(重写request)
  10. */
  11. public class RepeatableFilter implements Filter {
  12. @Override
  13. public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
  14. ServletRequest requestWrapper = null;
  15. //将原生的request变为可重复读取inputStream的request
  16. if (request instanceof HttpServletRequest
  17. && StringUtils.startsWithIgnoreCase(request.getContentType(), MediaType.APPLICATION_JSON_VALUE)) {
  18. requestWrapper = new RepeatedlyRequestWrapper((HttpServletRequest) request, response);
  19. }
  20. if (null == requestWrapper) {
  21. chain.doFilter(request, response);
  22. } else {
  23. chain.doFilter(requestWrapper, response);
  24. }
  25. }
  26. }

3.自定义注解

  1. /**
  2. * 自定义注解防止表单重复提交
  3. */
  4. @Inherited
  5. @Target(ElementType.METHOD)
  6. @Retention(RetentionPolicy.RUNTIME)
  7. @Documented
  8. public @interface RepeatSubmit {
  9. }

4.创建防止重复提交拦截器

  1. package com.hl.springbootrepetitionsubmit.interceptor;
  2. import cn.hutool.core.util.StrUtil;
  3. import com.alibaba.fastjson.JSONObject;
  4. import com.hl.springbootrepetitionsubmit.annotation.RepeatSubmit;
  5. import com.hl.springbootrepetitionsubmit.config.RedisUtil;
  6. import com.hl.springbootrepetitionsubmit.servlet.RepeatedlyRequestWrapper;
  7. import org.springframework.beans.factory.annotation.*;
  8. import org.springframework.stereotype.Component;
  9. import org.springframework.web.method.HandlerMethod;
  10. import org.springframework.web.servlet.HandlerInterceptor;
  11. import javax.servlet.http.*;
  12. import java.io.IOException;
  13. import java.lang.reflect.Method;
  14. import java.util.*;
  15. import java.util.concurrent.TimeUnit;
  16. /**
  17. * 防止重复提交拦截器
  18. */
  19. @Component
  20. public class RepeatSubmitInterceptor implements HandlerInterceptor {
  21. public final String REPEAT_PARAMS = "repeatParams";
  22. public final String REPEAT_TIME = "repeatTime";
  23. /**
  24. * 防重提交 redis key
  25. */
  26. public final String REPEAT_SUBMIT_KEY = "repeat_submit:";
  27. // 令牌自定义标识
  28. @Value("${token.header}")
  29. private String header;
  30. @Autowired
  31. private RedisUtil redisUtil;
  32. /**
  33. * 间隔时间,单位:秒
  34. * <p>
  35. * 两次相同参数的请求,如果间隔时间大于该参数,系统不会认定为重复提交的数据
  36. */
  37. @Value("${repeatSubmit.intervalTime}")
  38. private int intervalTime;
  39. @Override
  40. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
  41. if (handler instanceof HandlerMethod) {
  42. HandlerMethod handlerMethod = (HandlerMethod) handler;
  43. Method method = handlerMethod.getMethod();
  44. RepeatSubmit annotation = method.getAnnotation(RepeatSubmit.class);
  45. if (annotation != null) {
  46. if (this.isRepeatSubmit(request)) {
  47. //返回重复提交提示
  48. Map<String, Object> resultMap = new HashMap<>();
  49. resultMap.put("code", "500");
  50. resultMap.put("msg", request.getRequestURI() + "不允许重复提交,请稍后再试");
  51. try {
  52. response.setStatus(200);
  53. response.setContentType("application/json");
  54. response.setCharacterEncoding("utf-8");
  55. response.getWriter().print(JSONObject.toJSONString(resultMap));
  56. } catch (IOException e) {
  57. e.printStackTrace();
  58. }
  59. return false;
  60. }
  61. }
  62. return true;
  63. } else {
  64. return preHandle(request, response, handler);
  65. }
  66. }
  67. /**
  68. * 验证是否重复提交由子类实现具体的防重复提交的规则
  69. */
  70. public boolean isRepeatSubmit(HttpServletRequest request) {
  71. String nowParams = "";
  72. if (request instanceof RepeatedlyRequestWrapper) {
  73. RepeatedlyRequestWrapper repeatedlyRequest = (RepeatedlyRequestWrapper) request;
  74. nowParams = repeatedlyRequest.getBody();
  75. }
  76. // body参数为空,获取Parameter的数据
  77. if (StrUtil.isBlank(nowParams)) {
  78. nowParams = JSONObject.toJSONString(request.getParameterMap());
  79. }
  80. Map<String, Object> nowDataMap = new HashMap<String, Object>();
  81. nowDataMap.put(REPEAT_PARAMS, nowParams);
  82. nowDataMap.put(REPEAT_TIME, System.currentTimeMillis());
  83. // 请求地址(作为存放cache的key值)
  84. String url = request.getRequestURI();
  85. // 唯一值(没有消息头则使用请求地址)
  86. String submitKey = request.getHeader(header);
  87. if (StrUtil.isBlank(submitKey)) {
  88. submitKey = url;
  89. }
  90. // 唯一标识(指定key + 消息头)
  91. String cacheRepeatKey = REPEAT_SUBMIT_KEY + submitKey;
  92. Object sessionObj = redisUtil.getCacheObject(cacheRepeatKey);
  93. if (sessionObj != null) {
  94. Map<String, Object> sessionMap = (Map<String, Object>) sessionObj;
  95. if (sessionMap.containsKey(url)) {
  96. Map<String, Object> preDataMap = (Map<String, Object>) sessionMap.get(url);
  97. if (compareParams(nowDataMap, preDataMap) && compareTime(nowDataMap, preDataMap)) {
  98. return true;
  99. }
  100. }
  101. }
  102. Map<String, Object> cacheMap = new HashMap<String, Object>();
  103. cacheMap.put(url, nowDataMap);
  104. redisUtil.setCacheObject(cacheRepeatKey, cacheMap, intervalTime, TimeUnit.SECONDS);
  105. return false;
  106. }
  107. /**
  108. * 判断参数是否相同
  109. */
  110. private boolean compareParams(Map<String, Object> nowMap, Map<String, Object> preMap) {
  111. String nowParams = (String) nowMap.get(REPEAT_PARAMS);
  112. String preParams = (String) preMap.get(REPEAT_PARAMS);
  113. return nowParams.equals(preParams);
  114. }
  115. /**
  116. * 判断两次间隔时间
  117. */
  118. private boolean compareTime(Map<String, Object> nowMap, Map<String, Object> preMap) {
  119. long time1 = (Long) nowMap.get(REPEAT_TIME);
  120. long time2 = (Long) preMap.get(REPEAT_TIME);
  121. if ((time1 - time2) < (this.intervalTime * 1000L)) {
  122. return true;
  123. }
  124. return false;
  125. }
  126. }

5.redis配置

  1. package com.hl.springbootrepetitionsubmit.config;
  2. import org.springframework.cache.CacheManager;
  3. import org.springframework.cache.annotation.CachingConfigurerSupport;
  4. import org.springframework.cache.interceptor.KeyGenerator;
  5. import org.springframework.context.annotation.Bean;
  6. import org.springframework.context.annotation.Configuration;
  7. import org.springframework.data.redis.cache.RedisCacheConfiguration;
  8. import org.springframework.data.redis.cache.RedisCacheManager;
  9. import org.springframework.data.redis.connection.RedisStandaloneConfiguration;
  10. import org.springframework.data.redis.connection.jedis.JedisClientConfiguration;
  11. import org.springframework.data.redis.connection.jedis.JedisConnectionFactory;
  12. import org.springframework.data.redis.core.RedisTemplate;
  13. import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;
  14. import org.springframework.data.redis.serializer.RedisSerializationContext;
  15. import org.springframework.data.redis.serializer.StringRedisSerializer;
  16. import redis.clients.jedis.JedisPool;
  17. import redis.clients.jedis.JedisPoolConfig;
  18. import java.time.Duration;
  19. @Configuration
  20. public class RedisConfig extends CachingConfigurerSupport {
  21. //配置redis的过期时间
  22. private static final Duration TIME_TO_LIVE = Duration.ZERO;
  23. @Bean(name = "jedisPoolConfig")
  24. public JedisPoolConfig jedisPoolConfig() {
  25. JedisPoolConfig jedisPoolConfig = new JedisPoolConfig();
  26. //控制一个pool可分配多少个jedis实例
  27. jedisPoolConfig.setMaxTotal(500);
  28. //最大空闲数
  29. jedisPoolConfig.setMaxIdle(200);
  30. //每次释放连接的最大数目,默认是3
  31. jedisPoolConfig.setNumTestsPerEvictionRun(1024);
  32. //逐出扫描的时间间隔(毫秒) 如果为负数,则不运行逐出线程, 默认-1
  33. jedisPoolConfig.setTimeBetweenEvictionRunsMillis(30000);
  34. //连接的最小空闲时间 默认1800000毫秒(30分钟)
  35. jedisPoolConfig.setMinEvictableIdleTimeMillis(-1);
  36. jedisPoolConfig.setSoftMinEvictableIdleTimeMillis(10000);
  37. //最大建立连接等待时间。如果超过此时间将接到异常。设为-1表示无限制。
  38. jedisPoolConfig.setMaxWaitMillis(1500);
  39. jedisPoolConfig.setTestOnBorrow(true);
  40. jedisPoolConfig.setTestWhileIdle(true);
  41. jedisPoolConfig.setTestOnReturn(false);
  42. jedisPoolConfig.setJmxEnabled(true);
  43. jedisPoolConfig.setBlockWhenExhausted(false);
  44. return jedisPoolConfig;
  45. }
  46. @Bean("connectionFactory")
  47. public JedisConnectionFactory connectionFactory() {
  48. RedisStandaloneConfiguration redisStandaloneConfiguration = new RedisStandaloneConfiguration();
  49. redisStandaloneConfiguration.setHostName("127.0.0.1");
  50. redisStandaloneConfiguration.setDatabase(0);
  51. // redisStandaloneConfiguration.setPassword(RedisPassword.of("123456"));
  52. redisStandaloneConfiguration.setPort(6379);
  53. //获得默认的连接池构造器
  54. JedisClientConfiguration.JedisPoolingClientConfigurationBuilder jpcb =
  55. (JedisClientConfiguration.JedisPoolingClientConfigurationBuilder) JedisClientConfiguration.builder();
  56. //指定jedisPoolConifig来修改默认的连接池构造器(真麻烦,滥用设计模式!)
  57. jpcb.poolConfig(jedisPoolConfig());
  58. //通过构造器来构造jedis客户端配置
  59. JedisClientConfiguration jedisClientConfiguration = jpcb.build();
  60. //单机配置 + 客户端配置 = jedis连接工厂
  61. return new JedisConnectionFactory(redisStandaloneConfiguration, jedisClientConfiguration);
  62. }
  63. @Bean("jedisPool")
  64. public JedisPool jedisPool(){
  65. return new JedisPool(jedisPoolConfig(),"127.0.0.1",6379);
  66. }
  67. @Bean
  68. public RedisTemplate<Object, Object> redisTemplate(JedisConnectionFactory connectionFactory) {
  69. RedisTemplate<Object, Object> redisTemplate = new RedisTemplate<>();
  70. redisTemplate.setConnectionFactory(connectionFactory);
  71. GenericJackson2JsonRedisSerializer jackson2JsonRedisSerializer = new GenericJackson2JsonRedisSerializer();
  72. redisTemplate.setKeySerializer(new StringRedisSerializer());
  73. redisTemplate.setHashKeySerializer(new StringRedisSerializer());
  74. redisTemplate.setValueSerializer(jackson2JsonRedisSerializer);
  75. redisTemplate.setHashValueSerializer(jackson2JsonRedisSerializer);
  76. //初始化参数和初始化工作
  77. redisTemplate.afterPropertiesSet();
  78. return redisTemplate;
  79. }
  80. /**
  81. * 缓存对象集合中,缓存是以key-value形式保存的,
  82. * 当不指定缓存的key时,SpringBoot会使用keyGenerator生成Key。
  83. */
  84. @Override
  85. @Bean
  86. public KeyGenerator keyGenerator() {
  87. return (target, method, params) -> {
  88. StringBuilder sb = new StringBuilder();
  89. //类名+方法名
  90. sb.append(target.getClass().getName());
  91. sb.append(method.getName());
  92. for (Object obj : params) {
  93. sb.append(obj.toString());
  94. }
  95. return sb.toString();
  96. };
  97. }
  98. /**
  99. * 缓存管理器
  100. * @param connectionFactory 连接工厂
  101. * @return cacheManager
  102. */
  103. @Bean
  104. public CacheManager cacheManager(JedisConnectionFactory connectionFactory) {
  105. //新建一个Jackson2JsonRedis的redis存储的序列化方式
  106. GenericJackson2JsonRedisSerializer genericJackson2JsonRedisSerializer = new GenericJackson2JsonRedisSerializer();
  107. //解决查询缓存转换异常的问题
  108. // 配置序列化(解决乱码的问题)
  109. //entryTtl设置过期时间
  110. //serializeValuesWith设置redis存储的序列化方式
  111. RedisCacheConfiguration config = RedisCacheConfiguration.defaultCacheConfig()
  112. .entryTtl(TIME_TO_LIVE)
  113. .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(genericJackson2JsonRedisSerializer))
  114. .disableCachingNullValues();
  115. //定义要返回的redis缓存管理对象
  116. return RedisCacheManager.builder(connectionFactory).cacheDefaults(config).build();
  117. }
  118. }

6.redis工具类

  1. package com.hl.springbootrepetitionsubmit.config;
  2. import org.springframework.beans.factory.annotation.Autowired;
  3. import org.springframework.data.redis.core.BoundSetOperations;
  4. import org.springframework.data.redis.core.HashOperations;
  5. import org.springframework.data.redis.core.RedisTemplate;
  6. import org.springframework.data.redis.core.ValueOperations;
  7. import org.springframework.stereotype.Component;
  8. import java.util.*;
  9. import java.util.concurrent.TimeUnit;
  10. @Component
  11. public class RedisUtil {
  12. @Autowired
  13. private RedisTemplate redisTemplate;
  14. /**
  15. * 缓存基本的对象,Integer、String、实体类等
  16. *
  17. * @param key 缓存的键值
  18. * @param value 缓存的值
  19. */
  20. public <T> void setCacheObject(final String key, final T value) {
  21. redisTemplate.opsForValue().set(key, value);
  22. }
  23. /**
  24. * 缓存基本的对象,Integer、String、实体类等
  25. *
  26. * @param key 缓存的键值
  27. * @param value 缓存的值
  28. * @param timeout 时间
  29. * @param timeUnit 时间颗粒度
  30. */
  31. public <T> void setCacheObject(final String key, final T value, final Integer timeout, final TimeUnit timeUnit) {
  32. redisTemplate.opsForValue().set(key, value, timeout, timeUnit);
  33. }
  34. /**
  35. * 设置有效时间
  36. *
  37. * @param key Redis键
  38. * @param timeout 超时时间
  39. * @return true=设置成功;false=设置失败
  40. */
  41. public boolean expire(final String key, final long timeout) {
  42. return expire(key, timeout, TimeUnit.SECONDS);
  43. }
  44. /**
  45. * 设置有效时间
  46. *
  47. * @param key Redis键
  48. * @param timeout 超时时间
  49. * @param unit 时间单位
  50. * @return true=设置成功;false=设置失败
  51. */
  52. public boolean expire(final String key, final long timeout, final TimeUnit unit) {
  53. return redisTemplate.expire(key, timeout, unit);
  54. }
  55. /**
  56. * 获得缓存的基本对象。
  57. *
  58. * @param key 缓存键值
  59. * @return 缓存键值对应的数据
  60. */
  61. public <T> T getCacheObject(final String key) {
  62. ValueOperations<String, T> operation = redisTemplate.opsForValue();
  63. return operation.get(key);
  64. }
  65. /**
  66. * 删除单个对象
  67. *
  68. * @param key
  69. */
  70. public boolean deleteObject(final String key) {
  71. return redisTemplate.delete(key);
  72. }
  73. /**
  74. * 删除集合对象
  75. *
  76. * @param collection 多个对象
  77. * @return
  78. */
  79. public long deleteObject(final Collection collection) {
  80. return redisTemplate.delete(collection);
  81. }
  82. /**
  83. * 缓存List数据
  84. *
  85. * @param key 缓存的键值
  86. * @param dataList 待缓存的List数据
  87. * @return 缓存的对象
  88. */
  89. public <T> long setCacheList(final String key, final List<T> dataList) {
  90. Long count = redisTemplate.opsForList().rightPushAll(key, dataList);
  91. return count == null ? 0 : count;
  92. }
  93. /**
  94. * 获得缓存的list对象
  95. *
  96. * @param key 缓存的键值
  97. * @return 缓存键值对应的数据
  98. */
  99. public <T> List<T> getCacheList(final String key) {
  100. return redisTemplate.opsForList().range(key, 0, -1);
  101. }
  102. /**
  103. * 缓存Set
  104. *
  105. * @param key 缓存键值
  106. * @param dataSet 缓存的数据
  107. * @return 缓存数据的对象
  108. */
  109. public <T> BoundSetOperations<String, T> setCacheSet(final String key, final Set<T> dataSet) {
  110. BoundSetOperations<String, T> setOperation = redisTemplate.boundSetOps(key);
  111. Iterator<T> it = dataSet.iterator();
  112. while (it.hasNext()) {
  113. setOperation.add(it.next());
  114. }
  115. return setOperation;
  116. }
  117. /**
  118. * 获得缓存的set
  119. *
  120. * @param key
  121. * @return
  122. */
  123. public <T> Set<T> getCacheSet(final String key) {
  124. return redisTemplate.opsForSet().members(key);
  125. }
  126. /**
  127. * 缓存Map
  128. *
  129. * @param key
  130. * @param dataMap
  131. */
  132. public <T> void setCacheMap(final String key, final Map<String, T> dataMap) {
  133. if (dataMap != null) {
  134. redisTemplate.opsForHash().putAll(key, dataMap);
  135. }
  136. }
  137. /**
  138. * 获得缓存的Map
  139. *
  140. * @param key
  141. * @return
  142. */
  143. public <T> Map<String, T> getCacheMap(final String key) {
  144. return redisTemplate.opsForHash().entries(key);
  145. }
  146. /**
  147. * 往Hash中存入数据
  148. *
  149. * @param key Redis键
  150. * @param hKey Hash键
  151. * @param value 值
  152. */
  153. public <T> void setCacheMapValue(final String key, final String hKey, final T value) {
  154. redisTemplate.opsForHash().put(key, hKey, value);
  155. }
  156. /**
  157. * 获取Hash中的数据
  158. *
  159. * @param key Redis键
  160. * @param hKey Hash键
  161. * @return Hash中的对象
  162. */
  163. public <T> T getCacheMapValue(final String key, final String hKey) {
  164. HashOperations<String, String, T> opsForHash = redisTemplate.opsForHash();
  165. return opsForHash.get(key, hKey);
  166. }
  167. /**
  168. * 获取多个Hash中的数据
  169. *
  170. * @param key Redis键
  171. * @param hKeys Hash键集合
  172. * @return Hash对象集合
  173. */
  174. public <T> List<T> getMultiCacheMapValue(final String key, final Collection<Object> hKeys) {
  175. return redisTemplate.opsForHash().multiGet(key, hKeys);
  176. }
  177. /**
  178. * 获得缓存的基本对象列表
  179. *
  180. * @param pattern 字符串前缀
  181. * @return 对象列表
  182. */
  183. public Collection<String> keys(final String pattern) {
  184. return redisTemplate.keys(pattern);
  185. }
  186. }

7.配置拦截器

  1. /**
  2. * 防重复提交配置
  3. */
  4. @Configuration
  5. public class RepeatSubmitConfig implements WebMvcConfigurer {
  6. @Autowired
  7. private RepeatSubmitInterceptor repeatSubmitInterceptor;
  8. /**
  9. * 添加防重复提交拦截
  10. */
  11. @Override
  12. public void addInterceptors(InterceptorRegistry registry) {
  13. registry.addInterceptor(repeatSubmitInterceptor).addPathPatterns("/**");
  14. }
  15. /**
  16. * 生成防重复提交过滤器(重写request)
  17. * @return FilterRegistrationBean
  18. */
  19. @SuppressWarnings({ "rawtypes", "unchecked" })
  20. @Bean
  21. public FilterRegistrationBean<?> createRepeatableFilter() {
  22. FilterRegistrationBean registration = new FilterRegistrationBean();
  23. registration.setFilter(new RepeatableFilter());
  24. registration.addUrlPatterns("/*");
  25. registration.setName("repeatableFilter");
  26. registration.setOrder(FilterRegistrationBean.LOWEST_PRECEDENCE);
  27. return registration;
  28. }
  29. }

8.controller

  1. @RestController
  2. @RequestMapping("/repeatSubmit")
  3. public class RepeatSubmitController {
  4. /**
  5. * 保存Param
  6. * @param name
  7. * @return
  8. */
  9. @RepeatSubmit
  10. @PostMapping("/saveParam/{name}")
  11. public String saveParam(@PathVariable("name")String name){
  12. return "保存Param成功";
  13. }
  14. /**
  15. * 保存Param
  16. * @param name
  17. * @return
  18. */
  19. @RepeatSubmit
  20. @PostMapping("/saveBody")
  21. public String saveBody(@RequestBody List<String> name){
  22. return "保存Body成功";
  23. }
  24. }

三、测试

param传参:

body传参:

项目源码

相关文章