JSP+MySQL+Tomcat+Apache乱码解决一例

 

JSP+MySQL+Tomcat+Apache乱码解决一例

网摘精华

转自:http://www.matrix.org.cn/thread.shtml?topicId=25997&forumId=27
本文以最常见的JSP+MySQL+Tomcat+Apache乱码解决为例,望能为你的环境配置起到抛砖引玉之效!

  乱码问题已历来已久,在开源环境下,乱码问题更是令程序员措手不及。本人在Unix(Freebsd)下的一次乱码经历可谓经典,故撰以此文以滋效尤!

  我将本次所遇乱码归为三类:

  1.页面字符乱码

  2.记录显示乱码

  3.request传递乱码

  以下将对上述三类乱码进行解析:

一.页面字符乱码:

  1.大小写不一致:

org.apache.jasper.JasperException: /top.jsp(1,1) Page directive: illegal to have multiple occurrences of contentType with different values (old: text/html;charset=gb2312, new: text/html;charset=GB2312)

  2.间隔不一致:

org.apache.jasper.JasperException: /top.jsp(1,1) Page directive: illegal to have multiple occurrences of contentType with different values (old: text/html; charset=GB2312, new: text/html;charset=GB2312)

*解决方案:

首先,在Apache中增加AddDefaultCharset GB2312或AddDefaultCharset GBK

其次,统一使用页面编码定义,如:<%@page contentType="text/html;charset=GB2312"%>

*注:GB2312为GBK之子集。

二.记录显示乱码:

  1.MySQL默人语言为latin1_swedish_ci,即拉丁语,所以取出的中文全是乱码。

*解决方案:

  1.将charset设为8859_1即:<%@page contentType="text/html;charset=8859_1"%>

  这个方法只能暂时缓解字符显示问题,并权益之计。因为8859_1为字节型字库,并非字型字库,故在非全角状态下,将出现半字乱码,表现为“?”。

  2.在数据库连接语句中加上?useUnicode=true;characterEncoding=GBK,如:
jdbc:mysql://localhost/dbname?useUnicode=true;characterEncoding=GBK

*注:一般教科书上都会加上localhost:3306,因为默认端口为3306,故可舍去!同时,请使用连接池的朋友注意,在注册xml文件时,是不可以单独出现“;”的,所以必须使用“&”,即:jdbc:mysql://localhost/dbname?useUnicode=true&characterEncoding=GBK。

  否则提示出错:

Parse Fatal Error at line 213 column 91: The reference to entity "characterEncoding" must end with the ';' delimiter.
org.xml.sax.SAXParseException: The reference to entity "characterEncoding" must
end with the ';' delimiter.
at org.apache.xerces.util.ErrorHandlerWrapper.createSAXParseException(Un
known Source)

  也曾有人提意:在MySQL的my.ini文件中加入default-character-set=gbk,本人不赞同此法,因为这样破坏了原有的环境,除非这是MySQL的第一个站点。

三.request传递乱码:

  1.也许,此时你已经可以正常使用系统了,那么恭喜~乱码问题已经离开你了!但是,大伙通常都没那么走运,乱码问题依旧存在。也许,这时你向数据库添加了一条记录以测试系统,可是此时显示出的还是乱码。那么可以肯定是Request参数传递出错!那么先写个测试语句:<%= request.getParameter(“Para”) %>,OK,果然是乱。那么,现在有两种解决方法。

*解决方案:

  1.加上这条语句:request.setCharacterEncoding("gbk");
在一/两页上可行,但此法也非权益之计。

  2.注册SetCharacterEncodingFilter类:

  首先,编写SetCharacterEncodingFilter.java文件,代码如下:

package cn.com.jsp;

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;
import javax.servlet.UnavailableException;

public class SetCharacterEncodingFilter implements Filter {
protected String encoding = null;
protected FilterConfig filterConfig = null;
protected boolean ignore = true;

public void destroy() {
this.encoding = null;
this.filterConfig = null;
}

public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain) throws IOException,
ServletException {

// Conditionally select and set the character encoding to be used
if (ignore || (request.getCharacterEncoding() == null)) {
String encoding = selectEncoding(request);
if (encoding != null) {
request.setCharacterEncoding(encoding);
}
}

// Pass control on to the next filter
chain.doFilter(request, response);

}

public void init(FilterConfig filterConfig) throws ServletException {

this.filterConfig = filterConfig;
this.encoding = filterConfig.getInitParameter("encoding");
String value = filterConfig.getInitParameter("ignore");
if (value == null) {
this.ignore = true;
} else if (value.equalsIgnoreCase("true")) {
this.ignore = true;
} else if (value.equalsIgnoreCase("yes")) {
this.ignore = true;
} else {
this.ignore = false;
}

}

protected String selectEncoding(ServletRequest request) {
return (this.encoding);
}

}

  此文件为request过滤类,在全局编译前需进行注册。

  注册文件为:<%wwwroot%>/WEB-INF/web.xml。

  在此文件中加入如下代码即可:

<web-app>
<display-name>wwwroot</display-name>
<description>MySQL Test App</description>
<filter>
<filter-name>setCharacterEncodingFilter</filter-name>
<display-name>setCharacterEncodingFilter</display-name>
<description>setCharacterEncodingFilter</description>
<filter-class>cn.com.jsp.SetCharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>GBK</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>setCharacterEncodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
……
</web-app>

 OK,现在可以编译你的SetCharacterEncodingFilter.java文件啦!
至此,乱码将与你格格不入!

制作Flash时的奇遇利用指针计算字符串中指定字符出现次数[C语言]

Comments

avatar
Yong Gang Xu writes:

灵狼兄。

您的文章很管用。

用您的介绍的以下方法 1。 乱码问题解决了。既然您建议这个方法并非权益之计,故尝试方法2,结果总是显示乱码。
您能多指导一点儿么?

1.将charset设为8859_1即:<%@page contentType="text/html;charset=8859_1"%>

  这个方法只能暂时缓解字符显示问题,并权益之计。因为8859_1为字节型字库,并非字型字库,故在非全角状态下,将出现半字乱码,表现为“?”。

2.在数据库连接语句中加上?useUnicode=true;characterEncoding=GBK,如:
jdbc:mysql://localhost/dbname?useUnicode=true;characterEncoding=GBK

多谢。
------------------------------------------------------------------------------------------------------------------------------------------
-----------------------------------------------------自己曾经用过的过滤器--------CharsetTool.java----------------------
package jobs;
import java.io.UnsupportedEncodingException;
public class CharsetTool
{
    public CharsetTool() {
    }
public String getstring(String s)
{
if(s==null){s="";}
try
{
byte[] b=s.getBytes("gb2312");
s=new String(b);
}
catch(Exception e)
{
s="";
}
return s;
}
    public static String getISOToGBK(String str)
    {
        try
        {
            return new String(str.getBytes("iso-8859-1"), "gbk");
        }
        catch (UnsupportedEncodingException ex)
        {
            return null;
        }
    }
  public static String toiso(String str)
    {
        try
        {
            return new String(str.getBytes("gb2312"), "iso-8859-1");
        }
        catch (UnsupportedEncodingException ex)
        {
            return null;
        }
    }
    public static String getGBKToISO(String str)
    {
        try
        {
            return new String(str.getBytes("gbk"), "iso-8859-1");
        }
        catch (UnsupportedEncodingException ex)
        {
            return null;
        }
    }

    public static String getISOToUTF8(String str)
    {
        try
        {
            return new String(str.getBytes("iso-8859-1"), "utf-8");
        }
        catch (UnsupportedEncodingException ex)
        {
            return null;
        }
    }

    public static String getEncodingString(String str, String encoding)
    {
        String[] encodings = encoding.toLowerCase().replaceAll(" //s", "").split("//|");
        if (encodings.length == 2)
        {
            String encodingStr = "";
            try
            {
                encodingStr = new String(str.getBytes(encodings[0]), encodings[1]);
            }
            catch (UnsupportedEncodingException ex)
            {
            }
            finally
            {
                return encodingStr;
            }
        }
        return str;
    }
}
-------------------jsp----页面-----------
<jsp:useBean id="CharsetT" class="jobs.CharsetTool" scope="page"/>
String user_name=CharsetT.getISOToGBK(request.getParameter("user_name"));
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值