问题描述
由于项目需要,需要在代码中使用POST请求去调用另一个服务的接口,即不通过前端,A 项目直接在方法中发起HTTP请求调用 B 项目的接口,当请求体中的参数有中文时,参数接收后中文会变为“?”。
具体原因是参数的编码格式不对,默认为类型为 Content-Type: text/plain; charset=ISO-8859-1
,我们修改为 UTF-8 编码即可。
问题再现
A 服务中发起 HTTP 请求的接口:
@PostMapping("/testA")
public JSONObject testA(@RequestBody User user){
// 创建 CloseableHttpClient 对象
CloseableHttpClient httpClient = HttpClientBuilder.create().build();
// 构造请求体内容
try {
// 创建 HttpGet 请求
HttpPost httpPost = new HttpPost("http://localhost:8081/testB");
// 设置请求头,并指定接受的 Content-Type
httpPost.setHeader("Content-Type", "application/json");
StringEntity entity = new StringEntity(JSONObject.toJSONString(user));
// 设置请求体
httpPost.setEntity(entity);
// 发送请求并获取响应
HttpResponse response = httpClient.execute(httpPost);
// 从响应中获取实体内容,并返回
return (JSONObject) JSON.toJSON(EntityUtils.toString(response.getEntity()));
} catch (IOException e) {
e.printStackTrace();
}finally {
try {
httpClient.close();
} catch (IOException e) {
e.printStackTrace();
}
}
return null;
}
B 服务中的接口:
@PostMapping("/testB")
public JSONObject testB(@RequestBody User user){
System.out.println(user);
return (JSONObject) JSON.toJSON(user);
}
问题分析
-
当我们使用ApiFox进行调试时,请求的参数值传入中文,可以看到在 A 服务的接口中我们顺利的获得传来的数据,此时中文还没有乱码。
-
当我们使用
httpPost.setHeader("Content-Type", "application/json")
方式设置请求头,指定接收 Content-Type 时,org.apache.http.entity
会默认对其进行ISO-8859-1
编码,之后 HttpPost 会携带我们的参数向本地的 8081 端口发送 POST 请求。 -
B 服务的接口接收参数,可以看到中文变为了“?”。
解决方法
此时我们已经定位到了中文乱码问题是由于 A 服务对中文参数使用了错误的编码格式,从而导致 B 服务接收到的中文变为了 “?”,那么我们只需要在 A 服务中指定对参数的编码为 UTF-8 编码即可解决问题。同样的,我们可能还需要对 B 服务返回的数据指定 UTF-8 编码。修改代码如下:
-
在将请求体内容设置到 HttpEntity 时通过第二个参数
StandardCharsets.UTF_8
将其指定为 UTF-8 编码,再对 HttpPost 对象设置请求体。同理对从响应中获取的数据进行 UTF-8 编码。 -
可以看到 B 服务中可以顺利接收中文参数了。