zoukankan      html  css  js  c++  java
  • (转)XMLHTTP 乱码的解决方法(UTF8,GB2312 编码 解码)(附带解决DHTMLX不能用在GB2312页面编码下的中文问题)

    • 在数据发送一方,利用 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格式的。

     

     

  • 相关阅读:
    webpack 关于跨域的配置
    如何使用css变量
    样式重置
    vue+element_ui上传文件,并传递额外参数(自动上传)
    LeetCode-46-全排列
    LeetCode-39-组合总数
    LeetCode-33-搜索旋转排序数组
    LeetCode-207-课程表
    LeetCode-15-三数之和
    LeetCode-盛最多水的容器
  • 原文地址:https://www.cnblogs.com/panzhilei/p/1932161.html
Copyright © 2011-2022 走看看