解決tomcat下中文亂碼問(wèn)題
這些天開(kāi)發(fā)一個(gè)項(xiàng)目,,服務(wù)器是tomcat,操作系統(tǒng)是xp,采用的是MVC架構(gòu),,模式是采用 Facade模式,,總是出現(xiàn)亂碼,,自己也解決了好多天,同事也幫忙解決,,也參考了網(wǎng)上眾多網(wǎng)友的文章和意見(jiàn),,總算是搞定,。但是好記性不如爛筆桿,所以特意記下,,以防止自己遺忘,,同時(shí)也給那些遇到同樣問(wèn)題的人提供一個(gè)好的參考途徑:
(一) JSP頁(yè)面上是中文,但是看的是后是亂碼:
解決的辦法就是在JSP頁(yè)面的編碼的地方<%@ page language="java" contentType="text/html;charset=GBK" %>,,因?yàn)镴sp轉(zhuǎn)成Java文件時(shí)的編碼問(wèn)題,,默認(rèn)的話有的服務(wù)器是ISO-8859-1,如果一個(gè)JSP中直接輸入了中文,,Jsp把它當(dāng)作 ISO8859-1來(lái)處理是肯定有問(wèn)題的,,這一點(diǎn),我們可以通過(guò)查看Jasper所生成的Java中間文件來(lái)確認(rèn)
(二) 當(dāng)用Request對(duì)象獲取客戶(hù)提交的漢字代碼的時(shí)候,,會(huì)出現(xiàn)亂碼:
解決的辦法是:要配置一個(gè)filter,也就是一個(gè)Servelet的過(guò)濾器,,代碼如下:
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;
/**
* Example filter that sets the character encoding to be used in parsing the
* incoming request
*/
public class SetCharacterEncodingFilter implements Filter {
/**
* Take this filter out of service.
*/
public void destroy() {
}
/**
* Select and set (if specified) the character encoding to be used to
* interpret request parameters for this request.
*/
public void doFilter(ServletRequest request, ServletResponse response,
FilterChain chain)throws IOException, ServletException {
request.setCharacterEncoding("GBK");
// 傳遞控制到下一個(gè)過(guò)濾器
chain.doFilter(request, response);
}
public void init(FilterConfig filterConfig) throws ServletException {
}
}
配置web.xml
<filter>
<filter-name>Set Character Encoding</filter-name>
<filter-class>SetCharacterEncodingFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>Set Character Encoding</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
如果你的還是出現(xiàn)這種情況的話你就往下看看是不是你出現(xiàn)了第四中情況,你的Form提交的數(shù)據(jù)是不是用get提交的,,一般來(lái)說(shuō)用post提交的話是沒(méi)有問(wèn)題的,,如果是的話,你就看看第四中解決的辦法,。
還有就是對(duì)含有漢字字符的信息進(jìn)行處理,,處理的代碼是:
package dbJavaBean;
public class CodingConvert
{
public CodingConvert()
{
//
}
public String toGb(String uniStr){
String gbStr = "";
if(uniStr == null){
uniStr = "";
}
try{
byte[] tempByte = uniStr.getBytes("ISO8859_1");
gbStr = new String(tempByte,"GB2312");
}
catch(Exception ex){
}
return gbStr;
}
public String toUni(String gbStr){
String uniStr = "";
if(gbStr == null){
gbStr = "";
}
try{
byte[] tempByte = gbStr.getBytes("GB2312");
uniStr = new String(tempByte,"ISO8859_1");
}catch(Exception ex){
}
return uniStr;
}
}
你也可以在直接的轉(zhuǎn)換,首先你將獲取的字符串用ISO-8859-1進(jìn)行編碼,,然后將這個(gè)編碼存放到一個(gè)字節(jié)數(shù)組中,,然后將這個(gè)數(shù)組轉(zhuǎn)化成字符串對(duì)象就可以了,例如:
String str=request.getParameter(“girl”);
Byte B[]=str.getBytes(“ISO-8859-1”);
Str=new String(B);
通過(guò)上述轉(zhuǎn)換的話,,提交的任何信息都能正確的顯示,。
(三) 在Formget請(qǐng)求在服務(wù)端用request. getParameter(“name”)時(shí)返回的是亂碼;按tomcat的做法設(shè)置Filter也沒(méi)有用或者用 request.setCharacterEncoding("GBK");也不管用問(wèn)題是出在處理參數(shù)傳遞的方法上:如果在servlet中用 doGet(HttpServletRequest request, HttpServletResponse response)方法進(jìn)行處理的話前面即使是寫(xiě)了:
request.setCharacterEncoding("GBK");
response.setContentType("text/html;charset=GBK");
也是不起作用的,,返回的中文還是亂碼?。?!如果把這個(gè)函數(shù)改成doPost(HttpServletRequest request, HttpServletResponse response)一切就OK了,。
同樣,在用兩個(gè)JSP頁(yè)面處理表單輸入之所以能顯示中文是因?yàn)橛玫氖莗ost方法傳遞的,,改成get方法依舊不行。
由此可見(jiàn)在servlet中用doGet()方法或是在JSP中用get方法進(jìn)行處理要注意,。這畢竟涉及到要通過(guò)瀏覽器傳遞參數(shù)信息,,很有可能引起常用字符集的沖突或是不匹配。
解決的辦法是:
1) 打開(kāi)tomcat的server.xml文件,,找到區(qū)塊,,加入如下一行:
URIEncoding=”GBK”
完整的應(yīng)如下:
<Connector port="8080" maxThreads="150" minSpareThreads="25" maxSpareThreads="75" enableLookups="false" redirectPort="8443" acceptCount="100" debug="0" connectionTimeout="20000" disableUploadTimeout="true" URIEncoding="GBK"/>
2)重啟tomcat,一切OK,。
需要加入的原因大家可以去研究 $TOMCAT_HOME/webapps/tomcat-docs/config/http.html下的這個(gè)文件就可以知道原因了。需要注意的是:這個(gè)地方如果你要是用UTF-8的時(shí)候在傳遞的過(guò)程中在Tomcat中也是要出現(xiàn)亂碼的情況,,如果不行的話就換別的字符集,。
(四) JSP頁(yè)面上有中文,按鈕上面也有中文,,但是通過(guò)服務(wù)器查看頁(yè)面的時(shí)候出現(xiàn)亂碼:
解決的辦法是:首先在JSP文件中不應(yīng)該直接包含本地化的消息文本,,而是應(yīng)該通過(guò)<bean:message>標(biāo)簽從Resource Bundle中獲得文本。應(yīng)該把你的中文文本放到Application.properties文件中,,這個(gè)文件放在WEB-INF/classes/* 下,,例如我在頁(yè)面里有姓名,年齡兩個(gè)label,我首先就是要建一個(gè)Application.properties,,里面的內(nèi)容應(yīng)該是name=”姓名” age=”年齡”,然后我把這個(gè)文件放到WEB-INF/classes/properties/下,,接下來(lái)根據(jù) Application.properties文件,對(duì)他進(jìn)行編碼轉(zhuǎn)化,,創(chuàng)建一個(gè)中文資源文件,,假定名字是 Application_cn.properties。在JDK中提供了native2ascii命令,,他能夠?qū)崿F(xiàn)字符編碼的轉(zhuǎn)換,。在DOS環(huán)境中找到你放置Application.properties的這個(gè)文件的目錄,在DOS環(huán)境中執(zhí)行一下命令,,將生成按GBK編碼的中文資源文件 Application_cn.properties:native2ascii ?encoding gbk Application.properties Application_cn.properties執(zhí)行以上命令以后將生成如下內(nèi)容的Application_cn.properties文件: name=u59d3u540d age=u5e74u9f84,在Struts-config.xml中配置:<message-resources parameter="properties.Application_cn"/>,。到這一步,基本上完成了一大半,,接著你就要在JSP頁(yè)面上寫(xiě) <%@ page language="java" contentType="text/html;charset=GBK" %>,到名字的那個(gè)label是要寫(xiě)<bean:message key=”name”>,這樣的化在頁(yè)面上出現(xiàn)的時(shí)候就會(huì)出現(xiàn)中文的姓名,,年齡這個(gè)也是一樣,按鈕上漢字的處理也是同樣的,。
(五) 寫(xiě)入到數(shù)據(jù)庫(kù)是亂碼:
解決的方法:要配置一個(gè)filter,也就是一個(gè)Servelet的過(guò)濾器,,代碼如同第二種時(shí)候一樣。
如果你是通過(guò)JDBC直接鏈接數(shù)據(jù)庫(kù)的時(shí)候,,配置的代碼如下:jdbc:mysql://localhost:3306/workshopdb? useUnicode=true&characterEncoding=GBK,,這樣保證到數(shù)據(jù)庫(kù)中的代碼是不是亂碼。
如果你是通過(guò)數(shù)據(jù)源鏈接的化你不能按照這樣的寫(xiě)法了,,首先你就要寫(xiě)在配置文件中,,在tomcat 5.0.19中配置數(shù)據(jù)源的地方是在C:Tomcat 5.0confCatalinalocalhost這個(gè)下面,我建立的工程是workshop,,放置的目錄是webapp下面,workshop.xml 的配置文件如下:
<!-- insert this Context element into server.xml -->
<Context path="/workshop" docBase="workshop" debug="0"
reloadable="true" >
<Resource name="jdbc/WorkshopDB"
auth="Container"
type="javax.sql.DataSource" />
<ResourceParams name="jdbc/WorkshopDB">
<parameter>
<name>factory</name>
<value>org.apache.commons.dbcp.BasicDataSourceFactory</value>
</parameter>
<parameter>
<name>maxActive</name>
<value>100</value>
</parameter>
<parameter>
<name>maxIdle</name>
<value>30</value>
</parameter>
<parameter>
<name>maxWait</name>
<value>10000</value>
</parameter>
<parameter>
<name>username</name>
<value>root</value>
</parameter>
<parameter>
<name>password</name>
<value></value>
</parameter>
<!-- Class name for mm.mysql JDBC driver -->
<parameter>
<name>driverClassName</name>
<value>com.mysql.jdbc.Driver</value>
</parameter>
<parameter>
<name>url</name>
<value><![CDATA[jdbc:mysql://localhost:3306/workshopdb?useUnicode=true&characterEncoding=GBK]]></value>
</parameter>
</ResourceParams>
</Context>
粗體的地方要特別的注意,,和JDBC直接鏈接的時(shí)候是有區(qū)別的,如果你是配置正確的化,當(dāng)你輸入中文的時(shí)候到數(shù)據(jù)庫(kù)中就是中文了,,有一點(diǎn)要注意的是你在顯示數(shù)據(jù)的頁(yè)面也是要用<%@ page language="java" contentType="text/html;charset=GBK" %>這行代碼的,。需要注意的是有的前臺(tái)的人員在寫(xiě)代碼的是后用Dreamver寫(xiě)的,寫(xiě)了一個(gè)Form的時(shí)候把他改成了一個(gè)jsp,,這樣有一個(gè)地方要注意了,,那就是在Dreamver中Action的提交方式是request的,你需要把他該過(guò)來(lái),,因?yàn)樵趈sp的提交的過(guò)程中緊緊就是POST和 GET兩種方式,,但是這兩種方式提交的代碼在編碼方面還是有很大不同的,這個(gè)在后面的地方進(jìn)行說(shuō)明,。3
以上就是我在開(kāi)發(fā)系統(tǒng)中解決中文的問(wèn)題,,不知道能不能解決大家的問(wèn)題,時(shí)間匆忙,,沒(méi)有及時(shí)完善,,文筆也不是很好,有些地方估計(jì)是詞不達(dá)意,。大家可以給我意見(jiàn),,希望能共同進(jìn)步。