最近在开发过程中,遇到了我们公司的几个老项目,采用的是传统单体项目,这种项目之间需要交互,项目A请求项目B,B将结果回调给A,我们采用了Http Post请求的方式,在接收Http请求的时候,遇到了不少问题,所以总结一下接收Http Post请求的几种方式。
1.@RequestBody注解直接接收
请求类型必须是:application/json
/**
* 主动请求方-A
*/
@PostMapping("request-callback")
public static void requestCallback() {
TreeMap<String, String> map = new TreeMap<>();
map.put("name", "六六");
map.put("age", "2");
map.put("version", "1.0");
String response;
try {
response = RequestUtil.post("http://127.0.0.1:8080/callback/response-callback", map);
} catch (Exception e) {
throw new RuntimeException("请求异常!");
}
LOGGER.info("回调返回 response = :{}", response);
}
// 项目A日志结果:回调返回 response = :回调成功!
/**
* 接收请求方-B-1
*/
@PostMapping("response-callback")
public String responseCallback(@RequestBody CallbackDTO callbackDTO) {
if (callbackDTO != null) {
LOGGER.info("dto:{}", JSON.toJSONString(callbackDTO));
return "回调成功!";
}
return "回调失败!";
}
// 项目B日志结果:dto:{"age":2,"name":"六六","version":"1.0"}
2.WebUtils.getParametersStartingWith(request, “”) 表单方式接收
- 熟悉JavaWeb开发的同学应该都知道request中有一个方法叫做request.getParameterMap() ,返回一个Map;
比如,前端提交一个表单:
<input type="text" name="p_name" value="六六">
<input type="text" name="p_age" value="2">
<input type="text" name="f_sex" value="女">
request.getParameterMap()会得到一个Map:
{p_name=六六, p_age=2, f_sex=女}
但是我们后端可能只需要带"p_"这个前缀的字段,这时我们就可以用WebUtils.getParametersStartingWith(request, “p_”)这个方法去指定获取带"p_"前缀的字段,返回值也是一个Map。它会去掉前缀并且将不是"p_"前缀的字段过滤掉:
{name=六六, age=2}
也可以不传参,WebUtils.getParametersStartingWith(request, “”),他获取的结果和request.getParameterMap()一样:
{p_name=六六, p_age=2, f_sex=女}
- 同样的,如果请求类型是 “Content-Type”,“application/x-www-form-urlencoded” 时,也可以使用这个方法去接收;
/**
* 主动请求方-A
*/
@PostMapping("request-callback")
public static void requestCallback() {
TreeMap<String, String> map = new TreeMap<>();
map.put("p_name", "六六");
map.put("p_age", "2");
map.put("f_version", "1.0");
String response;
try {
response = RequestUtil.post("http://127.0.0.1:8080/callback/response-callback", map);
} catch (Exception e) {
throw new RuntimeException("请求异常!");
}
LOGGER.info("回调返回 response = :{}", response);
}
// 项目A日志结果:回调返回 response = :回调成功!
/**
* 接收请求方-B-2
*/
@PostMapping("response-callback")
public String responseCallback() {
Map<String, Object> params = WebUtils.getParametersStartingWith(request, "p_");
LOGGER.info("params :{}", JSON.toJSONString(params));
return "回调成功!";
}
// 项目B日志结果:params :{"age":"2","name":"六六"}
和我们预想的一样,项目B接收到了请求,获取到了"p_"前缀的参数,将前缀自动去掉,并且给了项目B回调结果。
3.request.getReader()的方式接收
/**
* 主动请求方-A
*/
@PostMapping("request-callback")
public static void requestCallback() {
TreeMap<String, String> map = new TreeMap<>();
map.put("name", "六六");
map.put("age", "2");
map.put("version", "1.0");
String response;
try {
response = RequestUtil.post("http://127.0.0.1:8080/callback/response-callback", map);
} catch (Exception e) {
throw new RuntimeException("请求异常!");
}
LOGGER.info("回调返回 response = :{}", response);
}
// 项目A日志结果:回调返回 response = :回调成功!
/**
* 接收请求方-B-3
*/
@PostMapping("response-callback")
public String responseCallback() {
String content;
try {
StringBuilder sb = new StringBuilder();
String line;
while (!StringUtils.isBlank((line = request.getReader().readLine()))) {
sb.append(line);
}
content = sb.toString();
LOGGER.info("content = {}", content);
} catch (Exception e) {
LOGGER.error("msg={},stack={}", e.getMessage(), ExceptionUtils.getStackTrace(e));
}
return "回调成功!";
}
// 项目B日志结果:content = {"age":"2","name":"六六","version":"1.0"}
使用request.getInputStream()返回请求内容字节流一样可以获取。
4.工具类
我使用的Http请求工具类有两个,一个是hutool工具类,一个自己封装的。
<!--hutool工具类-->
<dependency>
<groupId>cn.hutool</groupId>
<artifactId>hutool-all</artifactId>
<version>4.1.8</version>
</dependency>
/**
* fshows.com
* Copyright (C) 2013-2019 All Rights Reserved.
*/
package com.example.callbackdemo.utils;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.serializer.SerializerFeature;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.exception.ExceptionUtils;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.http.util.EntityUtils;
import java.io.IOException;
/**
* post请求工具类
* @author liuyuan
*/
@Slf4j
public class RequestUtils {
/**
* 超时时间10秒
*/
private static final int TIMEOUT = 10000;
public static String post(String url, Object object) {
RequestConfig requestConfig = RequestConfig.custom().setSocketTimeout(TIMEOUT).setConnectTimeout(TIMEOUT).build();
String content = null;
HttpPost post = new HttpPost(url);
post.addHeader("Content-Type", "application/json");
post.setConfig(requestConfig);
String body = JSON.toJSONString(object, SerializerFeature.WriteNullStringAsEmpty);
log.info("执行POST请求 body={}", body);
post.setEntity(new StringEntity(body, "UTF-8"));
try {
CloseableHttpClient client = HttpClientBuilder.create().build();
CloseableHttpResponse closeableHttpResponse = client.execute(post);
content = EntityUtils.toString(closeableHttpResponse.getEntity(), "UTF-8");
} catch (IOException e) {
log.warn("执行POST请求异常params={},e={}", JSON.toJSONString(object), ExceptionUtils.getStackTrace(e));
} finally {
post.releaseConnection();
}
return content;
}
}
5.总结
request.getParameter()、request.getInputStream()、request.getReader()和@RequestBody注解是互斥的,只能调用其中一个,如果想要同时使用,需要做特殊处理。