java中文乱码解决之道(8):解决URL中文乱码问题

我们主要通过两种形式提交向服务器发送请求:URL、表单。而表单形式一般都不会出现乱码问题,乱码问题主要是在URL上面。通过前面几篇博客的介绍我们知道URL向服务器发送请求编码过程实在是实在太混乱了。不同的操作系统、不同的浏览器、不同的网页字符集,将导致完全不同的编码结果。如果程序员要把每一种结果都考虑进去,是不是太恐怖了?有没有办法,能够保证客户端只用一种编码方法向服务器发出请求?
有!这里我主要提供以下几种方法
一、javascript
使用javascript编码不给浏览器插手的机会,编码之后再向服务器发送请求,然后在服务器中解码。在掌握该方法的时候,我们需要料及javascript编码的三个方法:escape()、encodeURI()、encodeURIComponent()。
escape
采用SIO Latin字符集对指定的字符串进行编码。所有非ASCII字符都会被编码为%xx格式的字符串,其中xx表示该字符在字符集中所对应的16进制数字。例如,格式对应的编码为%20。它对应的解码方法为unescape()。


事实上escape()不能直接用于URL编码,它的真正作用是返回一个字符的Unicode编码值。比如上面“我是cm”的结果为%u6211%u662Fcm,其中“我”对应的编码为6211,“是”的编码为662F,“cm”编码为cm。
注意,escape()不对”+”编码。但是我们知道,网页在提交表单的时候,如果有空格,则会被转化为+字符。服务器处理数据的时候,会把+号处理成空格。所以,使用的时候要小心。
encodeURI
对整个URL进行编码,它采用的是UTF-8格式输出编码后的字符串。不过encodeURI除了ASCII编码外对于一些特殊的字符也不会进行编码如:! @ # $& * ( ) = : / ; ? + ‘。
201501150003

encodeURIComponent()
把URI字符串采用UTF-8编码格式转化成escape格式的字符串。相对于encodeURI,encodeURIComponent会更加强大,它会对那些在encodeURI()中不被编码的符号(; / ? : @ & = + $ , #)统统会被编码。但是encodeURIComponent只会对URL的组成部分进行个别编码,而不用于对整个URL进行编码。对应解码函数方法decodeURIComponent。
当然我们一般都是使用encodeURI方来进行编码操作。所谓的javascript两次编码后台两次解码就是使用该方法。javascript解决该问题有一次转码、两次转码两种解决方法。
一次转码
javascript转码:
1
2

var url =
'<s:property value="webPath" />/ShowMoblieQRCode.servlet?name=我是cm'
;

window.location.href = encodeURI(url);

转码后的URL:http://127.0.0.1:8080/perbank/ShowMoblieQRCode.servlet?name=%E6%88%91%E6%98%AFcm
后台处理:
1
2
3
4

String name = request.getParameter(
"name"
);

System.out.println(
"前台传入参数:"

  • name);

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

System.out.println(
"经过解码后参数:"

  • name);

输出结果:
前台传入参数:??????cm经过解码后参数:我是cm
二次转码
javascript
1
2

var url =
'<s:property value="webPath" />/ShowMoblieQRCode.servlet?name=我是cm'
;

window.location.href = encodeURI(encodeURI(url));

转码后的url:http://127.0.0.1:8080/perbank/ShowMoblieQRCode.servlet?name=%25E6%2588%2591%25E6%2598%25AFcm
后台处理:
1
2
3
4

String name = request.getParameter(
"name"
);

System.out.println(
"前台传入参数:"

  • name);

name = URLDecoder.decode(name,
"UTF-8"
);

System.out.println(
"经过解码后参数:"

  • name);

输出结果:
前台传入参数:E68891E698AFcm
经过解码后参数:我是cm
filter
使用过滤器,过滤器LZ提供两种,第一种设置编码,第二种直接在过滤器中进行解码操作。
过滤器1
该过滤器是直接设置request的编码格式的。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

public
class
CharacterEncoding
implements
Filter {

private
FilterConfig config ;

String encoding =
null
;

public
void
destroy() {

config =
null
;

}

public
void
doFilter(ServletRequest request, ServletResponse response,

FilterChain chain)
throws
IOException, ServletException {

request.setCharacterEncoding(encoding);

chain.doFilter(request, response);

}

public
void
init(FilterConfig config)
throws
ServletException {

this
.config = config;

//获取配置参数

String str = config.getInitParameter(
"encoding"
);

if
(str!=
null
){

encoding = str;

}

}

}

配置:
1
2
3
4
5
6
7
8
9
10
11
12
13
14

<filter>

<filter-name>chineseEncoding</filter-name>

<filter-
class

com.test.filter.CharacterEncoding</filter-
class

<init-param>

<param-name>encoding</param-name>

<param-value>utf-
8
</param-value>

</init-param>

</filter>

<filter-mapping>

<filter-name>chineseEncoding</filter-name>

<url-pattern>/*</url-pattern>

</filter-mapping>

过滤器2
该过滤器在处理方法中将参数直接进行解码操作,然后将解码后的参数重新设置到request的attribute中。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77

public
class
CharacterEncoding
implements
Filter {

protected
FilterConfig filterConfig ;

String encoding =
null
;

public
void
destroy() {

this
.filterConfig =
null
;

}

/**

  • 初始化

*/

public
void
init(FilterConfig filterConfig) {

this
.filterConfig = filterConfig;

}

/**

  • 将 inStr 转为 UTF-8 的编码形式
  • @param inStr 输入字符串

  • @return UTF - 8 的编码形式的字符串
  • @throws UnsupportedEncodingException

*/

private
String toUTF(String inStr)
throws
UnsupportedEncodingException {

String outStr =
""
;

if
(inStr !=
null
) {

outStr =
new
String(inStr.getBytes(
"iso-8859-1"
),
"UTF-8"
);

}

return
outStr;

}

/**

  • 中文乱码过滤处理

*/

public
void
doFilter(ServletRequest servletRequest,

ServletResponse servletResponse, FilterChain chain)
throws
IOException,

ServletException {

HttpServletRequest request = (HttpServletRequest) servletRequest;

HttpServletResponse response = (HttpServletResponse) servletResponse;

// 获得请求的方式 (1.post or 2.get), 根据不同请求方式进行不同处理

String method = request.getMethod();

// 1. 以 post 方式提交的请求 , 直接设置编码为 UTF-8

if
(method.equalsIgnoreCase(
"post"
)) {

try
{

request.setCharacterEncoding(
"UTF-8"
);

}

catch
(UnsupportedEncodingException e) {

e.printStackTrace();

}

}

// 2. 以 get 方式提交的请求

else
{

// 取出客户提交的参数集

Enumeration<String> paramNames = request.getParameterNames();

// 遍历参数集取出每个参数的名称及值

while
(paramNames.hasMoreElements()) {

String name = paramNames.nextElement();
// 取出参数名称

String values[] = request.getParameterValues(name);
// 根据参数名称取出其值

// 如果参数值集不为空

if
(values !=
null
) {

// 遍历参数值集

for
(
int
i =
0
; i < values.length; i++) {

try
{

// 回圈依次将每个值调用 toUTF(values[i]) 方法转换参数值的字元编码

String vlustr = toUTF(values[i]);

values[i] = vlustr;

}

catch
(UnsupportedEncodingException e) {

e.printStackTrace();

}

}

// 将该值以属性的形式藏在 request

request.setAttribute(name, values);

}

}

}

// 设置响应方式和支持中文的字元集

response.setContentType(
"text/html;charset=UTF-8"
);

// 继续执行下一个 filter, 无一下个 filter 则执行请求

chain.doFilter(request, response);

}

}

配置:
1
2
3
4
5
6
7
8
9

<filter>

<filter-name>chineseEncoding</filter-name>

<filter-
class

com.test.filter.CharacterEncoding</filter-
class

</filter>

<filter-mapping>

<filter-name>chineseEncoding</filter-name>

<url-pattern>/*</url-pattern>

</filter-mapping>

其他
1、设置pageEncoding、contentType
1
2

<%@ page language=
"java"
contentType="text/html;

charset=UTF-
8
" pageEncoding="
UTF-
8
"%>

2、设置tomcat的URIEncoding
在默认情况下,tomcat服务器使用的是ISO-8859-1编码格式来编码的,URIEncoding参数对get请求的URL进行编码,所以我们只需要在tomcat的server.xml文件的<Connector>标签中加上URIEncoding=”utf-8″即可。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容