XMLHTTP 乱码的解决方法(UTF8,GB2312 编码 解码)


Posted in Javascript onJanuary 12, 2011

在数据发送一方,利用 javascript 的 escape 函数事先将所有中文转换成英文编码并保存(也可以用 ASP 等其它语言的函数实时转换输出)。
在数据接收一方,利用 javascript 的 unescape 函数将所有英文编码还原。
由于 Ajax 不论任何编码都能正常传送英文,所以用这种方法可以轻松解决。

--------------------------------------------------------------------------------
用XMLHTTP Post Form时的表单乱码有两方面的原因——Post表单数据时中文乱码;服务器Response被XMLHTTP不正确编码引起的乱码。换句话说,本文主要解决两个问题——怎样正确Post中文内容&怎样正确显示得到的中文内容。
Part I Post中文内容
先看看E文的表单是怎么提交的:

<SCRIPT language="JavaScript"> 
strA = "submit1=Submit&text1=scsdfsd"; 
var oReq = new ActiveXObject("MSXML2.XMLHTTP"); 
oReq.open("POST","http://ServerName/VDir/TstResult.asp",false); 
oReq.setRequestHeader("Content-Length",strA.length); 
oReq.setRequestHeader("CONTENT-TYPE","application/x-www-form-urlencoded"); 
oReq.send(strA); 
</ScRIPT>

如果把strA = "submit1=Submit&text1=scsdfsd";换成:
strA = "submit1=Submit&text1=中文";
你会发现提交上去的东东根本不对,ASP中Request.Form("Text1")根本取不到值。俺用Request.BinaryRead把一个HTML Form中的Post内容写出来看了看,才发现问题——Form提交时也要编码的,编码后的中文是类似于%??%??的转义字符,比如“中文”就被编码为:%D6%D0%CE%C4。呵呵,也怪俺笨,人家CONTENT-TYPE里明明写的清清楚楚——application/x-www-form-urlencoded,urlencoded嘛当然就是这个样子了。既然这样,那我们也知道该怎么办了——自己做转换,代码见下:
<SCRIPT language="VBScript"> 
Function URLEncoding(vstrIn) 
strReturn = "" 
For i = 1 To Len(vstrIn) 
ThisChr = Mid(vStrIn,i,1) 
If Abs(Asc(ThisChr)) < &HFF Then 
strReturn = strReturn & ThisChr 
Else 
innerCode = Asc(ThisChr) 
If innerCode < 0 Then 
innerCode = innerCode + &H10000 
End If 
Hight8 = (innerCode And &HFF00)\ &HFF 
Low8 = innerCode And &HFF 
strReturn = strReturn & "%" & Hex(Hight8) & "%" & Hex(Low8) 
End If 
Next 
URLEncoding = strReturn 
End Function 
strA = URLEncoding("submit1=Submit&text1=中文") 
oReq = CreateObject("MSXML2.XMLHTTP") 
oReq.open "POST","http://ServerName/VDir/TstResult.asp",false 
oReq.setRequestHeader "Content-Length",Len(strA) 
oReq.setRequestHeader "CONTENT-TYPE","application/x-www-form-urlencoded" 
oReq.send strA 
</SCRIPT>

(在这里俺把前面的JavaScript的代码改成了VBScript,不是吃饱了撑的没事干,原因见后)
Part II.正确显示得到的中文内容
OK,如果你在Server端把Form的内容写到数据库/文件的话,你在那里看到的中文毫无问题,但是,假如你想看看Server的Response——问题来了:如果Response的结果不是XML,XMLHTTP.responseXML里当然是不会有东东的,那就用responseText好了,在代码的最后加一句:
alert(oReq.responseText)
看看俺们辛勤劳动的结果 :P
但是但是.....怎么所有的中文全变成了方格? (我打不出来,有兴趣自己去试,也不用Post,Get一个含有中文的网页就可以发现了。)
原因很简单:XMLHTTP得到Response时假定Response是UTF8编码的,如果Response是XML,那还可以通过encoding来指定编码,但HTML就不行了。(见鬼的GB2312,再次打倒!)所以它把含GB2312编码的HTML当成UTF8格式,不出错才有鬼!
不过好在还有补救的办法:XMLHTTP的responseBody 属性里包含的可是未解码的Resonse——"a raw undecoded bytes as received directly from the server" :),唯一的问题是,responseBody返回的是一个unsigned bytes数组,我们怎么去访问它,怎么把它转换成BSTR?
这就是为什么我在上面把代码改成VBScript的原因——VBScript Can do it,but JavaScript Cannot!
代码见下:
<SCRIPT language="VBScript"> 
Function URLEncoding(vstrIn) 
strReturn = "" 
For i = 1 To Len(vstrIn) 
ThisChr = Mid(vStrIn,i,1) 
If Abs(Asc(ThisChr)) < &HFF Then 
strReturn = strReturn & ThisChr 
Else 
innerCode = Asc(ThisChr) 
If innerCode < 0 Then 
innerCode = innerCode + &H10000 
End If 
Hight8 = (innerCode And &HFF00)\ &HFF 
Low8 = innerCode And &HFF 
strReturn = strReturn & "%" & Hex(Hight8) & "%" & Hex(Low8) 
End If 
Next 
URLEncoding = strReturn 
End Function 
Function bytes2BSTR(vIn) 
strReturn = "" 
For i = 1 To LenB(vIn) 
ThisCharCode = AscB(MidB(vIn,i,1)) 
If ThisCharCode < &H80 Then 
strReturn = strReturn & Chr(ThisCharCode) 
Else 
NextCharCode = AscB(MidB(vIn,i+1,1)) 
strReturn = strReturn & Chr(CLng(ThisCharCode) * &H100 + CInt(NextCharCode)) 
i = i + 1 
End If 
Next 
bytes2BSTR = strReturn 
End Function 
strA = URLEncoding("submit1=Submit&text1=中文") 
oReq = CreateObject("MSXML2.XMLHTTP") 
oReq.open "POST","http://ServerName/VDir/TstResult.asp",false 
oReq.setRequestHeader "Content-Length",Len(strA) 
oReq.setRequestHeader "CONTENT-TYPE","application/x-www-form-urlencoded" 
oReq.send strA 
alert bytes2BSTR(oReq.responseBody) 
</SCRIPT>

----------------------------------------以上为转载-----------------------------
http://www.dhtmlx.com

Start Building Professional Ajax
Web Applications Today
最近用了 DHTMLX的tree,中文的xml居然用不了,报错,整个系统是GB2312的,而DHTMLX只支持UTF8,英文是正常的,中文就出错。

用法也很简单,是DHTMLX 演示里用的代码。

var obj=document.getElementById(oid); 
obj.style.display="block"; 
obj.focus(); 
if(obj.innerHTML!="")return; 
tree=new dhtmlXTreeObject(oid,"100%","100%",0); 
tree.setImagePath("js/imgs/csh_vista/"); 
tree.setOnClickHandler(click1); 
tree.setXMLAutoLoading("tree.asp"); 
//load first level of tree 
tree.loadXML("tree.asp?id=0");

tree.asp编码为gb2312,和整个系统一致,用response.write返回一个xml
整个系统编码改不了,只有在DHTMLX上想办法了。
查了些资料,最终解决了。

再深入一层发现是,Microsoft.XMLHTTP的原因,就查到了上面的文件。
如是就简单了,修改如下:
1. 增加一个VBScript的中文转换的函数

<script language="VBScript"> 
'处理中文 
function bin2str(bin) 
dim tmp,ustr,i 
tmp="" 
for i=1 to LenB(bin)-1 
ustr=AscB(MidB(bin,i,1)) 
if ustr>127 then 
i=i+1 
tmp=tmp&chr(ustr*256+AscB(MidB(bin,i,1))) 
else 
tmp=tmp&chr(ustr) 
end if 
next 
bin2str=tmp 
end function 
</script>

2.dhtmlxcommon.js的 getXMLTopNode处
var xmlString = this.xmlDoc.responseText;
改为
var xmlString = bin2str(this.xmlDoc.responseBody);//可以在js里直接调用VBScript脚本

3.同时 在输出xml的asp 文件开头加入
<%response.ContentType="application/xml"%>
<?xml version='1.0' encoding='GB2312'?>
当然文件也是保存为gb2312格式的。

Javascript 相关文章推荐
js DOM 元素ID就是全局变量
Sep 20 Javascript
jquery中show()、hide()和toggle()用法实例
Jan 15 Javascript
javascript日期验证之输入日期大于等于当前日期
Dec 13 Javascript
Javascript函数中的arguments.callee用法实例分析
Sep 16 Javascript
浅析js的模块化编写 require.js
Dec 07 Javascript
H5手机端多文件上传预览插件
Apr 21 Javascript
JavaScript实现QQ聊天消息展示和评论提交功能
May 22 Javascript
JS实现评价的星星功能
Aug 20 Javascript
js实现点击按钮复制文本功能
Jul 20 Javascript
三种Webpack打包方式(小结)
Sep 19 Javascript
vue给组件传递不同的值方法
Sep 29 Javascript
Vue向后台传数组数据,springboot接收vue传的数组数据实例
Nov 12 Javascript
奉献给JavaScript初学者的编写开发的七个细节
Jan 11 #Javascript
从盛大通行证上摘下来的身份证验证js代码
Jan 11 #Javascript
javascript 设为首页与加入收藏兼容多浏览器代码
Jan 11 #Javascript
javascript获取当前日期时间及其它操作函数
Jan 11 #Javascript
JavaScript中URL编码函数代码
Jan 11 #Javascript
Jquery跨域获得Json时invalid label错误的解决办法
Jan 11 #Javascript
JS无限树状列表实现代码
Jan 11 #Javascript
You might like
php MYSQL 数据备份类
2009/06/19 PHP
PHP 引用是个坏习惯
2010/03/12 PHP
zend api扩展的php对象的autoload工具
2011/04/18 PHP
PHP unlink与rmdir删除目录及目录下所有文件实例代码
2018/02/07 PHP
php ajax confirm 删除实例详解
2019/03/06 PHP
Laravel使用Queue队列的技巧汇总
2019/09/02 PHP
php array 转json及java 转换 json数据格式操作示例
2019/11/13 PHP
JS辨别访问浏览器判断是android还是ios系统
2014/08/19 Javascript
JAVASCRIPT代码编写俄罗斯方块网页版
2015/11/26 Javascript
vue短信验证性能优化如何写入localstorage中
2018/04/25 Javascript
详解Vue+axios+Node+express实现文件上传(用户头像上传)
2018/08/10 Javascript
原生JS+HTML5实现的可调节写字板功能示例
2018/08/30 Javascript
vue实现动态列表点击各行换色的方法
2018/09/13 Javascript
VUE的history模式下除了index外其他路由404报错解决办法
2019/08/21 Javascript
基于javascript实现贪吃蛇小游戏
2019/11/25 Javascript
Python学习笔记(一)(基础入门之环境搭建)
2014/06/05 Python
基于Django filter中用contains和icontains的区别(详解)
2017/12/12 Python
python实现微信跳一跳辅助工具步骤详解
2018/01/04 Python
Python中list查询及所需时间计算操作示例
2018/06/21 Python
对python3中pathlib库的Path类的使用详解
2018/10/14 Python
使用Python中的reduce()函数求积的实例
2019/06/28 Python
详解python 利用echarts画地图(热力图)(世界地图,省市地图,区县地图)
2019/08/06 Python
python动态视频下载器的实现方法
2019/09/16 Python
Selenium 滚动页面至元素可见的方法
2020/03/18 Python
CSS3等相关属性制作分页导航实现代码
2012/12/24 HTML / CSS
在线购买廉价折扣书籍和小说:BookOutlet.com
2018/02/19 全球购物
C语言中break与continue的区别
2012/07/12 面试题
家居设计专业个人自荐信范文
2013/11/26 职场文书
毕业生个人求职信范例分享
2013/12/17 职场文书
中学生期末评语
2014/02/03 职场文书
学校学雷锋活动总结
2014/06/26 职场文书
厨师长岗位职责范本
2014/08/25 职场文书
故宫的导游词
2015/01/31 职场文书
2015年感恩节演讲稿(优选篇)
2015/03/20 职场文书
干部考核工作总结2015
2015/07/24 职场文书
2016元旦文艺汇演主持词(开场白+结束语)
2015/12/03 职场文书