首页 > 技术文章 > request接收表单提交数据及其中文参数乱码问题

huangdabing 2018-06-14 01:45 原文

一、request接收表单提交数据:

 

  • getParameter(String)方法(常用)
  • getParameterValues(String name)方法(常用)
  • getParameterMap()方法(编写框架时常用)

 

比如现在有如下的form表单

 

复制代码
 1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
 2 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
 3 <html>
 4 <head>
 5     <title>Html的Form表单元素</title>
 6 </head>
 7 <fieldset style="width:500px;">
 8     <legend>Html的Form表单元素</legend>
 9     <!--form表单的action属性规定当提交表单时,向何处发送表单数据,method属性指明表单的提交方式,分为get和post,默认为get-->
10     <form action="${pageContext.request.contextPath}/servlet/RequestDemo03" method="post">
11     <!--输入文本框,SIZE表示显示长度,maxlength表示最多输入长度-->
12     编&nbsp;&nbsp;号(文本框):
13     <input type="text" name="userid" value="NO." size="2" maxlength="2"><br>
14     <!--输入文本框,通过value指定其显示的默认值-->
15     用户名(文本框):<input type="text" name="username" value="请输入用户名"><br>
16     <!--密码框,其中所有输入的内容都以密文的形式显示-->
17     密&nbsp;&nbsp;码(密码框):
18     <!--&nbsp;表示的是一个空格-->
19     <input type="password" name="userpass" value="请输入密码"><br>
20     <!--单选按钮,通过checked指定默认选中,名称必须一样,其中value为真正需要的内容-->
21     性&nbsp;&nbsp;别(单选框):
22     <input type="radio" name="sex" value="男" checked>23     <input type="radio" name="sex" value="女">女<br>
24     <!--下拉列表框,通过<option>元素指定下拉的选项-->
25     部&nbsp;&nbsp;门(下拉框):
26     <select name="dept">
27         <option value="技术部">技术部</option>
28         <option value="销售部" SELECTED>销售部</option>
29         <option value="财务部">财务部</option>
30     </select><br>
31     <!--复选框,可以同时选择多个选项,名称必须一样,其中value为真正需要的内容-->
32     兴&nbsp;&nbsp;趣(复选框): 
33     <input type="checkbox" name="inst" value="唱歌">唱歌 
34     <input type="checkbox" name="inst" value="游泳">游泳 
35     <input type="checkbox" name="inst" value="跳舞">跳舞 
36     <input type="checkbox" name="inst" value="编程" checked>编程 
37     <input type="checkbox" name="inst" value="上网">上网
38     <br>
39     <!--大文本输入框,宽度为34列,高度为5行-->
40     说&nbsp;&nbsp;明(文本域):
41     <textarea name="note" cols="34" rows="5">
42      </textarea>
43     <br>
44     <!--隐藏域,在页面上无法看到,专门用来传递参数或者保存参数-->
45     <input type="hidden" name="hiddenField" value="hiddenvalue"/>
46     <!--提交表单按钮,当点击提交后,所有填写的表单内容都会被传输到服务器端-->
47     <input type="submit" value="提交(提交按钮)">
48     <!--重置表单按钮,当点击重置后,所有表单恢复原始显示内容-->
49     <input type="reset" value="重置(重置按钮)">
50 </form>
51 <!--表单结束-->
52 </fieldset>
53 </body>
54 <!--完结标记-->
55 </html>
56 <!--完结标记-->
复制代码

 

  在Form表单中填写数据,然后提交到RequestDemo03这个Servlet进行处理,填写的表单数据如下:

 

  

 

在服务器端使用getParameter方法和getParameterValues方法接收表单参数,代码如下:

 

复制代码
 1 package gacl.request.study;
 2 import java.io.IOException;
 3 import java.text.MessageFormat;
 4 import javax.servlet.ServletException;
 5 import javax.servlet.http.HttpServlet;
 6 import javax.servlet.http.HttpServletRequest;
 7 import javax.servlet.http.HttpServletResponse;
 8 /**
 9  * @author gacl
10  * 获取客户端通过Form表单提交上来的参数
11  */
12 public class RequestDemo03 extends HttpServlet {
13 
14     public void doGet(HttpServletRequest request, HttpServletResponse response)
15             throws ServletException, IOException {
16         //客户端是以UTF-8编码提交表单数据的,所以需要设置服务器端以UTF-8的编码进行接收,否则对于中文数据就会产生乱码
17         request.setCharacterEncoding("UTF-8");
18         /**
19          * 编&nbsp;&nbsp;号(文本框):
20            <input type="text" name="userid" value="NO." size="2" maxlength="2">
21          */
22         String userid = request.getParameter("userid");//获取填写的编号,userid是文本框的名字,<input type="text" name="userid">
23         /**
24          * 用户名(文本框):<input type="text" name="username" value="请输入用户名">
25          */
26         String username = request.getParameter("username");//获取填写的用户名
27         /**
28          * 密&nbsp;&nbsp;码(密码框):<input type="password" name="userpass" value="请输入密码">
29          */
30         String userpass = request.getParameter("userpass");//获取填写的密码
31         String sex = request.getParameter("sex");//获取选中的性别
32         String dept = request.getParameter("dept");//获取选中的部门
33         //获取选中的兴趣,因为可以选中多个值,所以获取到的值是一个字符串数组,因此需要使用getParameterValues方法来获取
34         String[] insts = request.getParameterValues("inst");
35         String note = request.getParameter("note");//获取填写的说明信息
36         String hiddenField = request.getParameter("hiddenField");//获取隐藏域的内容
37         
38         String instStr="";
39         /**
40          * 获取数组数据的技巧,可以避免insts数组为null时引发的空指针异常错误!
41          */
42         for (int i = 0; insts!=null && i < insts.length; i++) {
43             if (i == insts.length-1) {
44                 instStr+=insts[i];
45             }else {
46                 instStr+=insts[i]+",";
47             }
48         }
49         
50         String htmlStr = "<table>" +
51                             "<tr><td>填写的编号:</td><td>{0}</td></tr>" +
52                             "<tr><td>填写的用户名:</td><td>{1}</td></tr>" +
53                             "<tr><td>填写的密码:</td><td>{2}</td></tr>" +
54                             "<tr><td>选中的性别:</td><td>{3}</td></tr>" +
55                             "<tr><td>选中的部门:</td><td>{4}</td></tr>" +
56                             "<tr><td>选中的兴趣:</td><td>{5}</td></tr>" +
57                             "<tr><td>填写的说明:</td><td>{6}</td></tr>" +
58                             "<tr><td>隐藏域的内容:</td><td>{7}</td></tr>" +
59                         "</table>";
60         htmlStr = MessageFormat.format(htmlStr, userid,username,userpass,sex,dept,instStr,note,hiddenField);
61         
62         response.setCharacterEncoding("UTF-8");//设置服务器端以UTF-8编码输出数据到客户端
63         response.setContentType("text/html;charset=UTF-8");//设置客户端浏览器以UTF-8编码解析数据
64         response.getWriter().write(htmlStr);//输出htmlStr里面的内容到客户端浏览器显示
65     }
66 
67     public void doPost(HttpServletRequest request, HttpServletResponse response)
68             throws ServletException, IOException {
69         doGet(request, response);
70     }
71 }
复制代码

 

运行结果如下:

 

  

在服务器端使用getParameterMap方法接收表单参数,代码如下:

复制代码
 1 //request对象封装的参数是以Map的形式存储的
 2         Map<String, String[]> paramMap = request.getParameterMap();
 3         for(Map.Entry<String, String[]> entry :paramMap.entrySet()){
 4             String paramName = entry.getKey();
 5             String paramValue = "";
 6             String[] paramValueArr = entry.getValue();
 7             for (int i = 0; paramValueArr!=null && i < paramValueArr.length; i++) {
 8                 if (i == paramValueArr.length-1) {
 9                     paramValue+=paramValueArr[i];
10                 }else {
11                     paramValue+=paramValueArr[i]+",";
12                 }
13             }
14             System.out.println(MessageFormat.format("{0}={1}", paramName,paramValue));
15         }
复制代码

运行结果如下:

  

二、request接收表单提交数据的中文参数乱码问题:

 

3.1、以POST方式提交表单中文参数的乱码问题

 

例如有如下的form表单页面:

 

复制代码
 1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
 2 
 3 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
 4 <html>
 5   <head>
 6     <title>request接收中文参数乱码问题</title>
 7   </head>
 8   
 9   <body>
10       <form action="<%=request.getContextPath()%>/servlet/RequestDemo04" method="post">
11           用户名:<input type="text" name="userName"/>
12           <input type="submit" value="post方式提交表单"> 
13       </form>
14   </body>
15 </html>
复制代码

 

  

 

  此时在服务器端接收中文参数时就会出现中文乱码,如下所示:

 

  

 

3.2、post方式提交中文数据乱码产生的原因和解决办法

 

  

 

  可以看到,之所以会产生乱码,就是因为服务器和客户端沟通的编码不一致造成的,因此解决的办法是:在客户端和服务器之间设置一个统一的编码,之后就按照此编码进行数据的传输和接收。

 

  由于客户端是以UTF-8字符编码将表单数据传输到服务器端的,因此服务器也需要设置以UTF-8字符编码进行接收,要想完成此操作,服务器可以直接使用从ServletRequest接口继承而来的"setCharacterEncoding(charset)"方法进行统一的编码设置。修改后的代码如下:

 

复制代码
1 public void doPost(HttpServletRequest request, HttpServletResponse response)
2             throws ServletException, IOException {
3         /**
4          * 客户端是以UTF-8编码传输数据到服务器端的,所以需要设置服务器端以UTF-8的编码进行接收,否则对于中文数据就会产生乱码
5          */
6         request.setCharacterEncoding("UTF-8");
7         String userName = request.getParameter("userName");
8         System.out.println("userName:"+userName);
9 }
复制代码

 

  使用request.setCharacterEncoding("UTF-8");设置服务器以UTF-8的编码接收数据后,此时就不会产生中文乱码问题了,如下所示:

 

  

 

3.3、以GET方式提交表单中文参数的乱码问题

 

例如有如下的form表单页面:

 

复制代码
 1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
 2 
 3 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
 4 <html>
 5   <head>
 6     <title>request接收中文参数乱码问题</title>
 7   </head>
 8   
 9   <body>
10         <form action="${pageContext.request.contextPath}/servlet/RequestDemo04" method="get">
11           姓名:<input type="text" name="name"/>
12           <input type="submit" value="get方式提交表单"> 
13       </form>
14   </body>
15 </html>
复制代码

 

  

 

  此时在服务器端接收中文参数时就会出现中文乱码,如下所示:

 

  

 

  那么这个中文乱码问题又该如何解决呢,是否可以通过request.setCharacterEncoding("UTF-8");设置服务器以UTF-8的编码进行接收这种方式来解决中文乱码问题呢,注意,对于以get方式传输的中文数据,通过request.setCharacterEncoding("UTF-8");这种方式是解决不了中文乱码问题,如下所示:

 

  

 

3.4、get方式提交中文数据乱码产生的原因和解决办法

 

  对于以get方式传输的数据,request即使设置了以指定的编码接收数据也是无效的(至于为什么无效我也没有弄明白),默认的还是使用ISO8859-1这个字符编码来接收数据,客户端以UTF-8的编码传输数据到服务器端,而服务器端的request对象使用的是ISO8859-1这个字符编码来接收数据,服务器和客户端沟通的编码不一致因此才会产生中文乱码的。解决办法:在接收到数据后,先获取request对象以ISO8859-1字符编码接收到的原始数据的字节数组,然后通过字节数组以指定的编码构建字符串,解决乱码问题。代码如下:

 

复制代码
 1 public void doGet(HttpServletRequest request, HttpServletResponse response)
 2             throws ServletException, IOException {
 3         /**
 4          *
 5          * 对于以get方式传输的数据,request即使设置了以指定的编码接收数据也是无效的,默认的还是使用ISO8859-1这个字符编码来接收数据
 6          */
 7         String name = request.getParameter("name");//接收数据
 8         name =new String(name.getBytes("ISO8859-1"), "UTF-8");//获取request对象以ISO8859-1字符编码接收到的原始数据的字节数组,然后通过字节数组以指定的编码构建字符串,解决乱码问题
 9         System.out.println("name:"+name);    
10 }
复制代码

 

运行结果如下:

 

 

3.5、以超链接形式传递中文参数的乱码问题

 

  客户端想传输数据到服务器,可以通过表单提交的形式,也可以通过超链接后面加参数的形式,例如:

 

1 <a href="${pageContext.request.contextPath}/servlet/RequestDemo05?userName=gacl&name=徐达沛">点击</a>

 

  点击超链接,数据是以get的方式传输到服务器的,所以接收中文数据时也会产生中文乱码问题,而解决中文乱码问题的方式与上述的以get方式提交表单中文数据乱码处理问题的方式一致,如下所示:

 

1 String name = request.getParameter("name");
2 name =new String(name.getBytes("ISO8859-1"), "UTF-8");

 

  另外,需要提的一点就是URL地址后面如果跟了中文数据,那么中文参数最好使用URL编码进行处理,如下所示:

 

1 <a href="${pageContext.request.contextPath}/servlet/RequestDemo05?userName=gacl&name=<%=URLEncoder.encode("徐达沛", "UTF-8")%>">点击</a>

 

3.6、提交中文数据乱码问题总结

 

  1、如果提交方式为post,想不乱码,只需要在服务器端设置request对象的编码即可,客户端以哪种编码提交的,服务器端的request对象就以对应的编码接收,比如客户端是以UTF-8编码提交的,那么服务器端request对象就以UTF-8编码接收(request.setCharacterEncoding("UTF-8"))

 

  2、如果提交方式为get,设置request对象的编码是无效的,request对象还是以默认的ISO8859-1编码接收数据,因此要想不乱码,只能在接收到数据后再手工转换,步骤如下:

 

  1).获取获取客户端提交上来的数据,得到的是乱码字符串,data="???è?????"

 

   String data = request.getParameter("paramName"); 

 

  2).查找ISO8859-1码表,得到客户机提交的原始数据的字节数组

 

   byte[] source = data.getBytes("ISO8859-1"); 

 

  3).通过字节数组以指定的编码构建字符串,解决乱码

 

   data = new String(source, "UTF-8"); 

 

       或

  String data = request.getParameter("paramName");

  data=new String(data.getBytes("ISO8859-1"), "UTF-8");

  通过字节数组以指定的编码构建字符串,这里指定的编码是根据客户端那边提交数据时使用的字符编码来定的,如果是GB2312,那么就设置成data = new String(source, "GB2312"),如果是UTF-8,那么就设置成data = new String(source, "UTF-8")

 

推荐阅读