SpringBoot使用@RequestParam和@RequestBody获取GET和POST请求

2 篇文章 0 订阅

前面两篇博客从客户端角度深入的分析了GET和POST的差异性和应用场景(从跨域问题探究Ajax的GET请求和地址栏发起GET请求的区别从使用角度分析GET和POST的差异性 应用场景)这篇博客就讲一下在服务端如何正确的处理GET和POST请求

spring boot中处理前端的GET和POST请求时,常用到的两个注解是

@RequestParam@RequestBody

@RequestParam:获取以&分割的数据,适用于获取一些比较少量的参数数据。

@RequestBody:获取请求体中的数据,适用于获取大量的数据。

具体使用得分场景,特别是post请求,有多样化的请求方式,需要对不同的方式用不同的注解。而且我看有的博客说@RequestParam处理请求头,@RequestBody处理请求体。这种说法不完全正确,因为@RequestParam也可以处理请求体中的数据,主要还是看参数的格式(也就是http的Content-Type)——是&分割的参数,还是key/value格式(application/x-www-form-urlencoded),又或是json格式(application/json)。

由于前端传递参数的多样化,这里细分几种常见的情况:

①前端使用GET请求,在地址栏传递数据,也就是说数据在请求的header中

http报文信息如下图

这种情况直接用@RequestParam接收数据即可。(这种情况不加注解也能获取到,但为了规范起见,最好加上),示例如下:

@GetMapping(value = "/getTest")
public String getTest(@RequestParam String param1,String param2){
    String res = "get传参已收到:"+param1+","+param2;
    return res;
}

②前端使用POST请求,通过application/x-www-form-urlencoded的方式传递数据(注意,form表单默认也是使用这种格式),会将表单的每个数据以key/value的形式传递,form表单的name属性对应key,form表单中的值对应value,这些key/value的数据存储是以xx=xx&xx=xx的方式在请求的body中。http报文信息如下图

所以这种情况也是用@RequestParam来处理,示例如下:

@PostMapping(value = "/postTest")
public String postTest(@RequestParam String param1,String param2){
    String res = "post传参已收到:"+param1+","+param2;
    return res;
}

如果不知道前端会传递多少个参数过来呢?或者传递的参数个数极多,对每个参数设置一个变量就显得几位麻烦。所以其实处理这种多个参数的请求,还有一种更加优雅的方式——使用一个map来接收传递的数据。

@PostMapping(value = "/postTest")
public String postTest(@RequestParam Map<String,String> map){
    StringBuilder res = new StringBuilder();
    res.append("post传参已收到:");
    map.forEach((k,v)->{//Lambda表达式写法,遍历map
        res.append(map.get(k)+"\t");
    });
    return res.toString();
}

这样就不必为每个参数一一对应去设置变量了,是不是极为方便?而且GET和POST都可以用使用这种方式。

如果前端传递的是一个类对象,还可以直接用一个类变量来代替map。比如public String postTest(@RequestParam User user)。

③前端使用POST请求,通过application/json的方式传递数据。传递的数据以json的格式存放在请求的body中,http报文信息如下图

这时候就需要用@RequestBody来处理,参数也是用一个map来接收。

@PostMapping(value = "/postTest")
public String postTest(@RequestBody Map<String,String> map){
    StringBuilder res = new StringBuilder();
    res.append("post传参已收到:");
    map.forEach((k,v)->{//Lambda表达式写法,遍历map
        res.append(map.get(k)+"\t");
    });
    return res.toString();
}

时间有限,暂时只罗列了这三种情况,后面遇到了其它情况再补充。

  • 24
    点赞
  • 48
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
实现接口调用频率限制可以使用AOP和ConcurrentHashMap结合的方式。 首先,在Spring Boot,我们可以使用AOP来拦截接口的调用。我们可以定义一个切面,使用@Aspect注解标注,然后在切入点方法定义需要拦截的注解。 例如,我们可以定义一个@FrequencyLimit注解,用于标注需要限制调用频率的方法: ```java @Retention(RetentionPolicy.RUNTIME) @Target({ ElementType.METHOD }) public @interface FrequencyLimit { // 限制时间段,单位为秒 int interval() default 60; // 时间段内最大请求次数 int maxCount() default 10; } ``` 然后,在切面,我们可以拦截该注解标注的方法,并且进行限制调用频率的操作。可以使用ConcurrentHashMap来存储每个接口的调用次数和最后一次调用时间。 ```java @Component @Aspect public class FrequencyLimitAspect { private ConcurrentHashMap<String, Long> lastRequestTimeMap = new ConcurrentHashMap<>(); private ConcurrentHashMap<String, Integer> requestCountMap = new ConcurrentHashMap<>(); @Around("@annotation(frequencyLimit)") public Object frequencyLimit(ProceedingJoinPoint joinPoint, FrequencyLimit frequencyLimit) throws Throwable { Object result = null; String methodName = joinPoint.getSignature().toLongString(); long currentTime = System.currentTimeMillis(); int interval = frequencyLimit.interval(); int maxCount = frequencyLimit.maxCount(); synchronized (this) { // 获取最后一次请求时间和请求次数 Long lastRequestTime = lastRequestTimeMap.get(methodName); Integer requestCount = requestCountMap.get(methodName); if (lastRequestTime == null || currentTime - lastRequestTime >= interval * 1000) { // 如果该接口在限制时间段内没有被调用过,则重置请求次数和最后一次请求时间 lastRequestTimeMap.put(methodName, currentTime); requestCountMap.put(methodName, 1); } else { // 如果该接口在限制时间段内已经被调用过,则增加请求次数 requestCount++; if (requestCount > maxCount) { // 如果请求次数超过了限制,则抛出异常 throw new RuntimeException("Exceeded maximum request limit"); } lastRequestTimeMap.put(methodName, currentTime); requestCountMap.put(methodName, requestCount); } } // 调用原始方法 result = joinPoint.proceed(); return result; } } ``` 在切面,我们使用synchronized关键字来保证线程安全,因为ConcurrentHashMap并不能完全保证线程安全。同时,我们使用了@Around注解来拦截被@FrequencyLimit注解标注的方法,然后在方法实现限制调用频率的逻辑。 这样,我们就可以实现接口调用频率限制了。在需要限制调用频率的方法,我们只需要加上@FrequencyLimit注解即可。例如: ```java @GetMapping("/test") @FrequencyLimit(interval = 60, maxCount = 10) public String test() { return "test"; } ``` 这样,每个IP地址每分钟内最多只能调用该方法10次,超过次数会抛出异常。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值