通过 AJAX 发布时字符编码错误

Posted

技术标签:

【中文标题】通过 AJAX 发布时字符编码错误【英文标题】:Bad character encoding when posting via AJAX 【发布时间】:2010-12-23 08:19:55 【问题描述】:

我的 Struts2 应用程序存在一些编码问题。我的 JSP 页面显示 2 个表单;当我发送第一个(一个简单的常规表单,它会重新加载整个页面)时,非标准字符,例如áñ 被发送并正确显示。但是,当我对第二种形式(通过 AJAX 发送)执行相同操作时,数据会损坏(ÃíÃ! 等等)。

ma​​in.jsp:

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<%@ page language="java" contentType="text/html;charset=UTF-8" pageEncoding="UTF-8" %>
<%@ taglib prefix="s" uri="/struts-tags" %>
<head>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
    ...
    <s:head theme="ajax"/>
</head>
<body>
    <!-- This form is sent via regular HTTP request -->
    <s:form theme="simple" enctype="multipart/form-data">
        <s:textfield key="field1" name="var1"/>
        <s:submit key="send" action="SAVE_ACTION"/>
    </s:form>
    <div id="ajaxContainer">
         <jsp:include file="ajax-part.jsp"/>
    </div>
</body>
</html>

ajax-part.jsp

<%@ page language="java" contentType="text/html;charset=UTF-8" pageEncoding="UTF-8" %>
<%@ taglib prefix="s" uri="/struts-tags" %>
<!-- This form is sent via AJAX -->
<s:form enctype="multipart/form-data" theme="ajax" action="AJAX_ACTION" acceptcharset="UTF-8">
    <s:textfield key="field2" name="var2"/>
    <s:submit key="send" targets="ajaxContainer"/>
</s:form>

如您所见,我在各处都设置了 UTF-8 编码。我还检查了这些文件实际上是 UTF-8 编码的。并且 DOJO(用于 AJAX 请求)也配置为 UTF-8,因为&lt;s:head theme="ajax"/&gt; 变为:

<script language="javascript" type="text/javascript">
    // Dojo configuration
    djConfig = 
        baseRelativePath: "/myApp/struts/dojo",
        isDebug: false,
        bindEncoding: "UTF-8",
        debugAtAllCosts: false
    ;
</script>

我在 Tomcat 6 服务器上运行应用程序;添加-Dfile.encoding=UTF-8 参数没有区别(除了以UTF-8 而不是ISO-8859-1 记录的日志文件)。

在谷歌浏览器和IE8中都有,所以我猜浏览器与问题无关,它一定是webapp中的东西。

我错过了什么?我可以更改哪些内容才能正确处理我的 AJAX 帖子?

【问题讨论】:

如果从
中删除 acceptcharset="UTF-8" 会发生什么?还尝试从 ajax 中删除 bindEncoding: "UTF-8" 吗?
没有 acceptcharset="UTF-8" 它似乎以同样的方式工作。至于DOJO配置的bindEncoding参数,不知道怎么去掉或者改.. Struts2 dojo 已被弃用(我忘了什么时候了,但到 2.2.1 肯定有)。我不认为它是服务器端的,因为您可以使用普通表单正确发布数据。使用相同的拦截器堆栈我猜是 dojo 是如何准备 url 的。也许手动管理 ajax(jQuery 和类似的 javascript 库)。 你用firebug检查过ajax请求了吗? 感谢四元数的 Firebug 提示。最后我能够解决这个问题:) 虽然我想要一个更好的解决方案...... 【参考方案1】:

解决了!

在使用 Firebug 检查 AJAX 请求后(感谢提示,四元数),我发现了问题:Dojo 在 GET 请求中正确发送数据,参数 URL 编码。首先,我认为 Struts2 使用错误的字符集进行 URL 解码,从而破坏了数据;但正如 Steven Benitez 所指出的,它实际上是底层的 Servlet API。

问题最终通过在 Tomcat 的 server.xml 中设置 URIEncoding="UTF-8"&lt;Connector&gt; 来解决,如 Apache Tomcat FAQs 中所述。

如果这种方法对其他人不起作用,也可以通过在接收 AJAX 数据的位置添加以下转换来手动解决

new String(myData.getBytes("ISO-8859-1"), "UTF-8");

【讨论】:

Struts2 不进行 URL 解码。底层的 Servlet API 可以。 ***.com/questions/469874/… 你的链接把我带到了这个页面wiki.apache.org/tomcat/FAQ/CharacterEncoding#Q2 - 我已经立即加入了书签。看来我终于找到了如何以正确的编码读取 GET 数据:)。我必须等到下周一才能测试它,但它应该可以工作。非常感谢!!

以上是关于通过 AJAX 发布时字符编码错误的主要内容,如果未能解决你的问题,请参考以下文章

为 AJAX 请求在 jQuery 中对字符串进行 URL 编码

是否有标准方法将 .NET 字符串编码为 JavaScript 字符串以在 MS Ajax 中使用?

“错误: 编码GBK的不可映射字符” 的解决方案

解决java编译错误:编码GBK的不可映射字符

Ajax 字符编码,进入 SQL 很奇怪

通过ajax从python返回unicode字符串