【Ajax技术】解决XHR与中文乱码问题

还是我们那个测试ajax的应用(服务端的Servlet、静态页面的html与实现ajax的javascript脚本文件)
服务端的Servlet

import java.io.IOException;
import java.io.PrintWriter;
import java.net.URLDecoder;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class AjaxServer extends HttpServlet {

	 protected void doPost(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws ServletException, IOException {
	        doGet(httpServletRequest, httpServletResponse);
	 }

	 protected void doGet(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws ServletException, IOException {
	        try{

	            httpServletResponse.setContentType("text/html;charset=utf-8");
	            PrintWriter out = httpServletResponse.getWriter();

	            //inte用来记录验证次数
	            Integer inte = (Integer) httpServletRequest.getSession().getAttribute("total");
	            int temp = 0;
	            if (inte == null) {
	                temp = 1;
	            } else {
	                temp = inte.intValue() + 1;
	            }
	            httpServletRequest.getSession().setAttribute("total",temp);

	            //1.取参数
	            String old = httpServletRequest.getParameter("name");
	            String name =old;
	            System.out.println(old+" oooooooooooooooooo");
	            //2.检查参数是否有问题
	            if(old == null || old.length() == 0){
	                out.println("用户名不能为空");
	            } else{

	                if(name.equals("hpu")){
	                    //4。和传统应用不同之处。这一步需要将用户感兴趣的数据返回给页面段,而不是将一个新的页面发送给用户
	                    //写法没有变化,本质发生了改变
	                    out.println("用户名[" + name + "]已经存在,请使用其他用户名, " + temp);
	                } else{
	                    out.println("用户名[" + name + "]尚未存在,可以使用该用户名注册, " + temp);
	                }
	            }
	        } catch(Exception e){
	            e.printStackTrace();
	       }
	 }
}

静态页面

ajaxXml.html:

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <title>用户校验ajax实例</title>
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
    <script type="text/javascript" src="js/jquery-1.8.0.js"></script>
    <script type="text/javascript" src="js/verify.js"></script>
  </head>

  <body>
    <h1> 用户校验ajax实例</h1><br>
    请输入用户名:<br/>
    <!-- ajax不需要使用表单进行数据提交,因此不用写表单标签 -->

    <!-- ajax不需要name属性,只需要一个id的属性 -->
    <input type="text" id="username"/>
    <input type="button" value="校验" onclick="verify()"/>
    <!-- 这个div用于存放服务器返回的信息,开始为空 -->
    <!-- id属性定义是为了利用dom的方式找到某一个节点,进行操作 -->
    <div id="result">

    </div>

  </body>
</html>

verify.js:

function verify(){
	var jqueryObj=$("#username");
	var userName=jqueryObj.val();
	$.get("AjaxServer?name="+userName,null,callback);

}

function callback(data){
	var resultObj=$("#result");
	resultObj.html(data);
}

当我们在静态页面端输入123的时候,返回的是:
用户名[123]尚未存在,可以使用该用户名注册, 1

当我们输入"中"的时候,返回的是:
用户名[??-]尚未存在,可以使用该用户名注册, 2

说明出现了中文乱码问题

剖析:
我们填写"中"之后,我们发现发送请求的url路径变成了:
http://localhost:8080/AjaxTrain/AjaxServer?name=%E4%B8%AD
它是按照UTF-8的方式转换为字节信息,一个“中”字,它按utf-8编码是3个字节。
(按照utf-8编码是一般浏览器指定的,如果在浏览器设置中取消,那么就会按照操作系统默认的编码方式去编码)
我们在Servlet中并没有将浏览器传过来的字节请求转化为字符数据并重新编码,所以导致了这个结果。

解决方法1(老方法):
在服务器端转换编码方式:
String name = new String(old.getBytes("iso8859-1"),"UTF-8");
这样转换可以的原因是tomcat服务器默认使用iso8859-1去编码得到的字节流变成字符流,这样我们将字符流以iso8859-1的方式解码为字节流,然后将解码后的字节流重新转码为UTF-8格式,就可以恢复中文。

在servlet加上这句代码后,输入"中"之后结果为:
用户名[中]尚未存在,可以使用该用户名注册, 1
说明转化成功。

如果仔细的话,可以看到,一些IE浏览器还会发生乱码情况。
我们还要专门为了部分IE浏览器去解决这个问题。
办法:
在页面显示的指定编码方式之外,在javascript脚本中对url进行编码:
verify.js:

function verify(){
	var jqueryObj=$("#username");
	var userName=encodeURI(jqueryObj.val());
	$.get("AjaxServer?name="+userName,null,callback);
}

function callback(data){
	var resultObj=$("#result");
	resultObj.html(data);
}

但是上面不是最完美的解决方法,因为我们在
String name = new String(old.getBytes("iso8859-1"),"UTF-8");中
iso8859-1是写死的,虽然各种服务器默认的编码方式是iso8859-1,但是一些
服务器人员会通过配置文件去修改它的默认编码方式,这个时候,我们就要去
修改我们的代码,非常不灵活,代码试用性不强。

解决方法2:
页面发出的数据做两次encodeURI:
verify.js:

function verify(){
	var jqueryObj=$("#username");
	var userName=encodeURI(encodeURI(jqueryObj.val()));
	$.get("AjaxServer?name="+userName,null,callback);
}

function callback(data){
	var resultObj=$("#result");
	resultObj.html(data);
}

服务器端:

String name=URLDecoder.decode(old,"UTF-8");

输入中文测试成功(包括IE)!

内部原理:
我们利用调试方式,去看看在js脚本文件中对url信息两次encodeURI有什么区别:
没做encodeURI:中
第一次encodeURI:%E4%B8%AD
第二次%E4%B8%AD:%25E4%25B8%25AD

第一次encodeURI把中这个数组分解成3个字节数组,每一个字节都对应一个16进制的表示法,
然后在他们面前加一个%。第二次encodeURI就会把前面的%转换成25再加上%。

而服务器的解码是如何进行?
我们给服务器的信息是%25E4%25B8%25AD,服务器正常解码之后的信息是%E4%B8%AD。(因为都是ASCII码中的字符,怎么解码都不会错)。之后我们再在我们的Servlet中使用Java的URLDecoder按照UTF-8的编码方式进行解码,就得到了最原始的中文信息。
转载请注明出处:http://blog.csdn.net/acmman/article/details/47755723

时间: 2024-08-31 19:42:20

【Ajax技术】解决XHR与中文乱码问题的相关文章

jquery.ajax的url中传递中文乱码问题的解决方法

 本篇文章主要是对jquery.ajax的url中传递中文乱码问题的解决方法进行了介绍,需要的朋友可以过来参考下,希望对大家有所帮助 JQuery   JQuery默认的contentType:application/x-www-form-urlencoded    这才是JQuery正在乱码的原因,在未指定字符集的时候,是使用ISO-8859-1    ISO8859-1,通常叫做Latin-1.Latin-1包括了书写所有西方欧洲语言不可缺少的附加字符.   JQuery的Ajax根本没有考

解决Ajax加载JSon数据中文乱码问题

一.问题描述 使用zTree的异步刷新父级菜单时,服务器返回中文乱码,但项目中使用了SpringMvc,已经对中文乱码处理,为什么还会出现呢? 此处为的异步请求的配置: Java代码 async: { enable: true, url: basePath + '/sysMenu/listSysMenu', autoParam: ["id=parentId"] } SpringMvc中文字符处理: Java代码 <mvc:annotation-driven> <mvc

彻底解决Spring MVC中文乱码问题的方案_java

乱码是让人很头疼的一件事,本文介绍了彻底解决Spring MVC中文乱码问题的方案,具体如下:  1:表单提交controller获得中文参数后乱码解决方案 注意:  jsp页面编码设置为UTF-8 form表单提交方式为必须为post,get方式下面spring编码过滤器不起效果 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <form

解决COOKIES存储中文乱码的问题

<%=Tsingda.XXB.Common.Utils.UrlDecode(i.KeyWord)%> 这其实是一个基础知识的问题,我们在获取URL地址参数时,在设置或者获取时最好将它进行编码,这样就不会有问题了,例如: Standard.ClientHelper.AddSearchRecord(Common.Utils.UrlEncode(_keyword), Convert.ToInt32(_module)); 在页面显示时只需要这样: <%=Tsingda.XXB.Common.Ut

解决ajax提交表单时中文乱码的问题

网上搜索结合自己开发过程中遇到的情况,整理一下,以备忘: Ajax表单提交数据出现乱码和解决办法    //要传递的参数   var queryString = "firstName=" + firstName + "&lastName=" + lastName                          + "&birthday=" + birthday;    function   //GET方式提交    doRequ

JQuery ajax 返回json时出现中文乱码该如何解决_AJAX相关

使用jQuery ajax调用的返回json,中文乱码问题 Js代码如下: $.ajax({ url: '/test/testAction.do?method=test', type: 'POST', dataType: 'json', timeout: 5000, async: false, error: function(){ alert('获取数据失败!'); }, success: function(json){ jsObject = eval(json); } }); return j

JQuery ajax 返回json时出现中文乱码该如何解决

使用jQuery ajax调用的返回json,中文乱码问题 Js代码如下: $.ajax({ url: '/test/testAction.do?method=test', type: 'POST', dataType: 'json', timeout: 5000, async: false, error: function(){ alert('获取数据失败!'); }, success: function(json){ jsObject = eval(json); } }); return j

设置正确的Content-Type以解决Ext的中文乱码问题

1.前后台所有文件统一用utf-8编码方式. 2.在Request Headers中设置Content-Type:application/x-www-form-urlencoded; charset=utf-8,具体方法:将Ext.lib.Ajax.defaultPostHeader += '; charset=utf-8'加在Ext.onReady块里,此方法可以解决用EXT在POST时的中文乱码问题. 3.在Response Headers中设置Content-Type:text/json;

如何解决Eclipse的中文乱码问题

使用Eclipse编辑文件经常出现中文乱码或者文件中有中文不能保存的问题,Eclipse提供了灵活的设置文件编码格式的选项,我们可以通过设置编码 格式解决乱码问题.在Eclipse可以从几个层面设置编码格式:Workspace.Project.Content Type.File 本文以Eclipse 3.3(英文)为例加以说明: 1. 设置Workspace的编码格式: Windows->Preferences... 打开"首选项"窗口,点击左侧导航树到General->W