一、项目配置:

  1. Spring 4.4.1-RELEASE
  2. Jetty 9.3.5
  3. JDK 1.8
  4. Servlet 3.1.0
  5. web.xml文件中没有配置编解码Filter

二、实际遇到的问题:
客户端(比如java)发送post请求访问接口,数据放在body里面,每个参数utf-8编码。
从body里面取出的中文参数是乱码。


下面是发送请求的代码和服务端接收请求的代码。

  • 客户端代码。
    这是一个真实的第三方访问API的案例,这段代码请求到PHP系统正常,请求到java系统就会出现乱码。
    但是中文参数放到URL中解码正常,放到请求体中就是乱码。
    通过httpclient4.1发送Post请求如下:

    public static void postData(String sign, String timestamp) {    // 创建默认的httpClient实例.
    CloseableHttpClient httpclient=null;
    String result="";
    try {
    httpclient = HttpClients.createDefault();
    String url = "http://example/api/entry";
    HttpPost httpPost = new HttpPost(url); //设置请求和传输超时时间
    RequestConfig requestConfig = RequestConfig.custom().setSocketTimeout(6000).setConnectTimeout(6000).build();
    httpPost.setConfig(requestConfig);
    MultipartEntity entity = new MultipartEntity(HttpMultipartMode.BROWSER_COMPATIBLE);
    entity.addPart("app_id", new StringBody("c5eb3ba8c0e7326559", Charset.forName("utf-8")));
    entity.addPart("method", new StringBody("kdt.item.add", Charset.forName("utf-8")));
    entity.addPart("timestamp", new StringBody(timestamp));
    entity.addPart("format", new StringBody("json", Charset.forName("utf-8")));
    entity.addPart("v", new StringBody("1.0", Charset.forName("utf-8")));
    entity.addPart("sign", new StringBody(sign, Charset.forName("utf-8")));
    entity.addPart("sign_method", new StringBody("md5", Charset.forName("utf-8")));
    entity.addPart("cid", new StringBody("5000000", Charset.forName("utf-8")));
    entity.addPart("tag_ids", new StringBody("0", Charset.forName("utf-8")));
    entity.addPart("price", new StringBody("0.01", Charset.forName("utf-8")));
    entity.addPart("title", new StringBody("测试", Charset.forName("utf-8")));
    entity.addPart("desc", new StringBody("test1", Charset.forName("utf-8"))); //是否是虚拟商品。0为否,1为是。目前不支持虚拟商品
    entity.addPart("is_virtual", new StringBody("0", Charset.forName("utf-8")));
    entity.addPart("post_fee", new StringBody("0.0", Charset.forName("utf-8"))); //Sku的属性串。格式:pText:vText;pText:vText,多个sku之间用逗号分隔,如:颜色:黄色;尺寸:M,颜色:黄色;尺寸:S。pText和vText文本中不可以存在冒号和分号以及逗号
    entity.addPart("sku_properties", new StringBody("color:white", Charset.forName("utf-8")));
    entity.addPart("sku_quantities", new StringBody("998,999", Charset.forName("utf-8")));
    entity.addPart("sku_prices", new StringBody("0.01,0.02", Charset.forName("utf-8")));
    entity.addPart("sku_outer_ids", new StringBody("null,null", Charset.forName("utf-8"))); //该商品的外部购买地址。当用户购买环境不支持微信或微博支付时会跳转到此地址
    entity.addPart("buy_url", new StringBody("http://img.cdn.sb.hongware.com/1461836641703511.gif", Charset.forName("utf-8")));
    entity.addPart("quantity", new StringBody("1998", Charset.forName("utf-8"))); //宝贝修改的时候需要这个参数
    httpPost.setEntity(entity);
    CloseableHttpResponse response = httpclient.execute(httpPost);
    try {
    HttpEntity httpEntity = response.getEntity();
    System.out.println(httpEntity.getContent());
    InputStream content = httpEntity.getContent();
    BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(content));
    String line;
    while ( (line=bufferedReader.readLine()) != null) {
    System.out.println(line);
    }
    } catch (Exception e) {
    e.printStackTrace();
    } finally {
    response.close();
    }
    } catch (Exception e) {
    e.printStackTrace();
    } finally {
    try {
    httpclient.close();
    } catch (Exception e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
    }
    }
    }
  • 服务端代码如下
    为了简化演示,我把参数提取代码Map<String, String[]> parameterMap = request.getParameterMap()冗余在这个入口函数中,以便说明问题:

    @RequestMapping(value = "/api/entry", produces = "application/json;charset=utf-8")
    public DeferredResult<Object> sign(HttpServletRequest request,HttpServletResponse response) {
    DeferredResult<Object> deferredResult = new DeferredResult<>();
    Map<String, String[]> parameterMap = request.getParameterMap();
    String method = request.getParameter("method");
    if (StringUtils.isEmpty(method)) {
    ResponseEntity<String> responseEntity = new ResponseEntity<String>( String.format(Constants.ERROR_RESPONSE, 50000, "service or method is null"), HttpStatus.valueOf(200));
    deferredResult.setResult(responseEntity);
    return deferredResult;
    }
    int lastIndex = method.lastIndexOf(".");
    String service = method.substring(0, lastIndex);
    method = method.substring(lastIndex + 1);
    event.setService(service);
    event.setMethod(method);
    event.setResult(deferredResult);
    proxy.doAction(request,response,event);
    return deferredResult;
    }

服务端通过Map<String, String[]> parameterMap = request.getParameterMap()取出所有参数,传进来title参数是乱码!!

三、根本原因
Servlet 3.0规范中有关请求数据编码的解释如下:

当前很多浏览器并不发送带Content-Type头部的字符编码标识符,它会把字符编码的决定留在读取HTTP请求的时候。如果客户端没有指明编码,容器用来创建请求读和解析POST数据的默认编码必须是"ISO-8859-1"。然而,为了提示开发者客户端没有成功发送一个字符编码,容器中getCharacterEncoding方法会返回null。
如果客户端没有设置字符编码,并且请求数据使用了不同编码而不是上述的默认编码,程序将会出现中断。为了纠正这种状态,一个新的方法setCharacterEncoding(String enc) 被添加到ServletRequest接口。开发者调用这个方法能重写容器提供的字符编码。这个方法必须在解析request中任何post数据或者读任何输入之前调用。一旦数据已经被读取,调用这个方法不会影响它的编码。

另外一种相同的解释:

网络上同样含义的解释

四、3种解决方法

  1. 在web.xml中配置编解码Filter

     <filter>
    <filter-name>encodingFilter</filter-name>
    <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
    <init-param>
    <param-name>encoding</param-name>
    <param-value>UTF-8</param-value>
    </init-param>
    <init-param>
    <param-name>forceEncoding</param-name>
    <param-value>true</param-value>
    </init-param>
    <async-supported>true</async-supported>
    </filter>
    <filter-mapping>
    <filter-name>encodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
    </filter-mapping>

    关于这段配置需要强调两点:

    • web.xml中,这段配置要放在所有filter的最前面,否则会不生效,根本原因请见上述第三点的解释。
    • 两个初始化参数的作用,其实看这个Filter的源码就一目了然,这两个参数是用来决定是否要设置request和response中的编码。源码很简洁:
       public class CharacterEncodingFilter extends OncePerRequestFilter {
      private String encoding;
      private boolean forceEncoding = false;
      public CharacterEncodingFilter() { }
      public void setEncoding(String encoding) {
      this.encoding = encoding;
      }
      public void setForceEncoding(boolean forceEncoding) {
      this.forceEncoding = forceEncoding;
      }
      protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
      if(this.encoding != null && (this.forceEncoding || request.getCharacterEncoding() == null)) {
      request.setCharacterEncoding(this.encoding);
      if(this.forceEncoding) {
      response.setCharacterEncoding(this.encoding);
      }
      }
      filterChain.doFilter(request, response);
      }
      }
  2. 设置Content-Type
    如果post请求方式是x-www-form-urlencoded,那么设置如下:
    Content-Type=application/x-www-form-urlencoded;charset=utf-8
    这样通过request对象取body体里面的中文是正常的。
    这种方式有一点需要注意: 如果请求方式是multipart/form-data,如上设置会导致request取不到参数。Content-Type要与传递数据匹配(本文data)
  3. 手动编解码
    比如参数title="测试",这样取出来就是"测试"。
     String str = new String(request.getParameter("title").getBytes("iso-8859-1"), "utf-8");

综上所有,最优雅的方式是第一种解决方案--通过框架的Filter去处理。
你仅专注于业务代码就好。

参考资料

    1. ajax post data获取不到数据
    2. Servlet 3.0规范
    3. HTTP Content-Type常用对照表
    4. Spring官网--Consumable Media Types章节
    5. ISO-8859-1
    6. ISO-8859-1为何能显示中文
    7. 字符编码
    8. Media Type

Spring MVC的Post请求参数中文乱码的原因&处理的更多相关文章

  1. spring mvc 文件下载 get请求解决中文乱码问题

    方案简写,自己或有些基础的可以看懂,因为没时间写的那么详细 方案1 spring mvc解决get请求中文乱码问题, 在tamcat中server.xml文件 URIEncoding="UT ...

  2. java web项目get,post请求参数中文乱码解决

    [转载]原文地址:https://www.cnblogs.com/tom-plus/p/6392279.html 在开发过程中,有时候会碰到get,post请求参数中文乱码. 原因: Http请求传输 ...

  3. 使用过滤器(Filter)解决请求参数中文乱码问题(复杂方式)

    前述:      在写这篇笔记之前,对笔记中的设计模式进行介绍:      本篇笔记中将要使用到的设计模式是:装饰(包装)设计模式           (1)装饰(包装)设计模式口诀:         ...

  4. 详解get请求和post请求参数中文乱码的解决办法

    首先出现中文乱码的原因是tomcat默认的编码方式是"ISO-8859-1",这种编码方式以单个字节作为一个字符,而汉字是以两个字节表示一个字符的. 一,get请求参数中文乱码的解 ...

  5. get请求和post请求参数中文乱码的解决办法

    get请求参数中文乱码的解决办法 在tomcat的server.xml里的Connector加个URIEncoding="UTF-8",把 <Connector connec ...

  6. 解决jmeter 请求参数中文乱码

    今天在用jmeter 写脚本时发现查看结果树request post请求中文参数值是乱码,故记录下解决过程. 解决过程如下: 1.修改本地配置文件 因为此处的数据,还没有发送出去,所以,肯定是这个变量 ...

  7. JSP中解决获取请求参数中文乱码问题

    分两种情况: 1.获取访问请求参数时乱码 解决方法:构造一个新的String String user = new String(request.getParameter("user" ...

  8. SpringMVC 接收表单数据、数据绑定、解决请求参数中文乱码

    接收表单数据有3种方式. 1.使用简单类型接收表单数据(绑定简单数据类型) 表单: <form action="${pageContext.request.contextPath}/u ...

  9. Spring MVC 结合Velocity视图出现中文乱码的解决方案

    编码问题一直是个很令人头疼的事,这几天搭了一个Spring MVC+VTL的web框架,发现中文乱码了,这里记录一种解决乱码的方案. 开发环境为eclipse,首先,检查Window->pref ...

随机推荐

  1. LaTeX技巧205:使用split输入多行公式技巧

    我们在输入多行公式的时候,split,array,multiline,align,aligned等等都是我们可以选用的环境,这里介绍split的使用方法.演示效果图: 演示代码:\documentcl ...

  2. mac或者linux磁力下载方法:远离渣雷

    wget是linux下常用的命令行下载工具,是Linux用户是必不可少的工具,尤其对于网络管理员,经常要下载一些软件. t-get是一个简单的命令行BT下载工具,可以用于BT种子和磁力链接的下载 tg ...

  3. Understanding Linux CPU stats

    Your Linux server is running slow, so you follow standard procedure and run top. You see the CPU met ...

  4. java问题排查命令

    java问题排查命令 jps:查看java进程 jmap:导出堆详细信息(与jhat一起使用) jhat:分析Java堆的命令(与jmap一起使用) jstack:可以定位到线程堆栈,根据堆栈信息我们 ...

  5. iOS开发-仿大众点评iPad侧边导航栏

    昨天其实已经写了一篇侧边栏的文章,不过感觉还不是很清晰,这篇文章算是补充吧,iPad上看了大众点评的侧边栏,基本上百分之九十类似,具体效果可参考下图: 对比昨天主要做了两个修改,一个是图片和文字的显示 ...

  6. Sum Root to Leaf Numbers leetcode java

    题目: Given a binary tree containing digits from 0-9 only, each root-to-leaf path could represent a nu ...

  7. UEditor常用设置函数记录

    在线文档对UEditor说明不够全面,收集了一些常用的方法和基本设置,以供参考. 1.创建编辑器 UE.getEditor('editor', { initialFrameWidth:"10 ...

  8. Android -- Toolbar跟随ListView滑动隐藏和显现

    布局 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:too ...

  9. sklearn文本特征提取

    http://cloga.info/2014/01/19/sklearn_text_feature_extraction/ 文本特征提取 词袋(Bag of Words)表征 文本分析是机器学习算法的 ...

  10. dubbo Framework pic

    dubbo Framework pic