相信大家在进行web开发时经常会遇到中文处理的困惑。我们经常会发现web应用在处理英文时很正常,但是处理中文时却出现乱码现象。最郁闷的是部分乱码,部分又显示正常,搞得是不知所措。下面以我在一个实际项目中的经验,介绍如何在struts中处理中文。我采用的中文编码是utf8,相信对于gbk编码方式也大同小异。
1. 数据库表编码
那数据库表时采用utf8编码。如下:
CREATE TABLE `project` ( `project_id` INTEGER(10) NOT NULL AUTO_INCREMENT, `project_name` VARCHAR(30) COLLATE gbk_chinese_ci DEFAULT NULL, `group_id` INTEGER(10) DEFAULT NULL, `responser` VARCHAR(30) COLLATE gbk_chinese_ci DEFAULT NULL, `status_id` INTEGER(10) NOT NULL, `description` VARCHAR(5000) COLLATE gbk_chinese_ci DEFAULT NULL, PRIMARY KEY (`project_id`) )ENGINE=MyISAM AUTO_INCREMENT=7 CHARACTER SET 'utf8' COLLATE 'utf8_general_ci';
2.jdbc连数据库的url中要给出编码方式,否则jdbc在给数据库写中文时就写入了乱码。只有在url给出编码方式时jdbc才能正确地将中文写入到数据库中。对于utf8编码的数据库连接url如下:
mysqlUrl=jdbc:mysql://127.0.0.1:3306/pmp?characterEncoding=utf8
3.jsp页面编码需要采用utf8
<%@ page language="java" pageEncoding="UTF-8"%> <%@ taglib uri="http://jakarta.apache.org/struts/tags-bean" prefix="bean"%> <%@ taglib uri="http://jakarta.apache.org/struts/tags-html" prefix="html"%> <%@ taglib uri="/WEB-INF/struts-logic.tld" prefix="logic"%> <html> <head> ...
4.struts中加入处理中文编码的filter,这样会把action中得到的request内容进行utf8编码.
filter的代码如下:
package com.mycompany.mypackage; import java.io.IOException; import javax.servlet.Filter; import javax.servlet.FilterChain; import javax.servlet.FilterConfig; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; public class UTF8Filter implements Filter { public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { request.setCharacterEncoding("UTF-8"); chain.doFilter(request, response); } public void destroy() { // do nothing } public void init(FilterConfig arg0) throws ServletException { // do nothing } }配置struts的filter:
在web.xml中加入以下黑体部分代码:
<web-app xmlns="http://java.sun.com/xml/ns/j2ee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" version="2.4" xsi:schemaLocation="http://java.sun.com/xml/ns/j2ee http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd"> <filter> <filter-name>UTF8Filter</filter-name> <filter-class>com.mycompany.mypackage .UTF8Filter</filter-class> </filter> <filter-mapping> <filter-name>UTF8Filter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping> <servlet>通过以上步骤,各个流程中的中文处理就ok了。
------------------------------------------------------------------------------------------------------------------
5. 可是还发现了一个奇怪的现象,对于下载文件功能的中文文件名还是显示乱码。对于服务器上文件的保存,我是通过在数据库中建一张表,记录了文件在服务器的名字,以及它在客户端的名称。在服务器上的文件名是随机生成的唯一文件名,这样客户是可以上传同名文件。所以在下载文件时通过文件表中的id查到文件在服务器上的名称和客户端文件名称,通过服务器上的名称去对应的文件夹下载,但是给客户端的名称是表中存放的客户端文件名称。对于struts中上传下载文件的处理见另外的blog"struts中上传下载文件的处理". 以下加黑部分是处理中文文件名的代码,主要是需要将文件名转换成iso字符集。
public class FileDownloadServlet extends HttpServlet { private ServletConfig config; public void init(ServletConfig config) throws ServletException { this.config = config; } public ServletConfig getServletConfig() { return config; } public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.setContentType("text/html; charset=UTF-8"); SmartUpload mySmartUpload = new SmartUpload(); String clientSideName = 从数据库中读到clientSideName; clientSideName = convertToIso(clientSideName); mySmartUpload.downloadFile("configure".equals(type) ? FileUtil.CONFIGURE_FILES_PATH + serverSideName : FileUtil.TESTRESULT_FILES_PATH + serverSideName, null, clientSideName); } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } // 将字符转换成ISO // 强制将字符转换成ISO public String convertToIso(String s) { try { return new String(s.getBytes("GBK"),"iso-8859-1"); } catch (UnsupportedEncodingException e) { // TODO Auto-generated catch block e.printStackTrace(); } return "wrong name"; } }
ok,通过以上方法就解决了我在构建这个struts应用时所磁到的中文乱码问题。
相关资源:struts2最新中文文档(高清)