分享web开发知识

注册/登录|最近发布|今日推荐

主页 IT知识网页技术软件开发前端开发代码编程运营维护技术分享教程案例
当前位置:首页 > 技术分享

关于乱码问题的解决与HttpServletResponse中的方法

发布时间:2023-09-06 01:23责任编辑:彭小芳关键词:乱码乱码问题

关于乱码问题的解决

会有乱码现象,其实就是因为字符集编码不一致的问题,就好像中国人和外国人谈话一样,互相不懂对方在说啥。字符集编码也是如此,本来就是一段GBK编码的文字,却要用utf-8的编码格式去解码,就当然是鸡同鸭讲会出现乱码啦,这个时候就得使用GBK编码的格式去解码才不会出问题。如果互相都是使用的GBK编码后,那就像中国人和中国人都说普通话一样,就能听懂对方在说什么,这样才不会出现乱码。

在web开发中,请求或响应数据时出现乱码,往往就是客户端和服务端的编码不一致的问题所导致的。

不过在介绍如何解决乱码的问题前,我们先看看HttpServletRequest中关于获得表单数据的一些方法,虽然在上一篇也介绍了使用方式,不过关于乱码和拿到具体的值这方面没有涉及到:


获得和设置表单数据方法(如果是上传文件的话则无法获取文件中的数据):

方法名称

作用

setCharacterEncoding(String)

设置提交上来表单的文本编码

getParameter(String)

得到表单中某一个指定的name属性的值

getParameterMap()

获得所有的键值对(name/value)

getParameterNames()

获得所有的name属性的值:

getParameterValues(String)

获得重复的name属性的值


既然和表单有关,那么就得先写一个简单的html表单代码,我们可以在Eclipse中创建一个html文件:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE53C172B578934E5DA89555F031242A14/4594" alt="4594" />

650) this.width=650;" style="width:533px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICED3E3B819FF6340A29A4A33F6E1D178B8/4595" alt="4595" />


可能使用Eclipse编写HTML的代码不太方便,我们也可以使用一个专门编写html代码的工具来编写Eclipse里已经创建了的html文件,我这里使用HBuilder作为示例:

  1. 复制Eclipse中的html文件所在目录的路径:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE028DD8ACEF9248FBA7EDAB648448A2BD/4596" alt="4596" />


  1. 在HBuilder中点击文件,然后选择打开目录把复制的文件路径粘贴进去,并为这个工程起一个新的名称:

650) this.width=650;" style="width:387px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEB50F0FBAA4054D3E8D5DF933D39E6133/4597" alt="4597" />

650) this.width=650;" style="width:524px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE6D80531D93954940A52E69ADB069591A/4598" alt="4598" />


工程目录如下:

650) this.width=650;" style="width:265px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE9FDE6A5F658541238B8732D903C5B900/4599" alt="4599" />

如图,可以看到index.html已经在这个工程下了,我们可以在HBuilder中编辑这个html文件,编辑的内容会同步到Eclipse,因为它俩访问的都是同一个目录同一个html文件。


  1. 我在HBuilder编辑的代码如下:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEF0C8F39A4B8D4E6C8F12A3811609E313/4600" alt="4600" />


  1. 再看看Eclipse发生了什么:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEAC67B89F586C4F5C925B4FD7345BD47A/4601" alt="4601" />

可以看到代码是同步的。


浏览器运行结果:

650) this.width=650;" style="width:337px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE6051C4310DD54F9AB1EAA91326C38BF2/4602" alt="4602" />


以下使用实际代码演示常用的几个获得表单数据的方法,代码示例:

650) this.width=650;" style="width:552px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE5A706944DC2F4AC2AFB71453386D7271/4603" alt="4603" />

650) this.width=650;" style="width:552px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE089CCDFF90244598B36437ECD789503D/4605" alt="4605" />


在Eclipse中执行html文件,Eclipse有一个内置的浏览器:

650) this.width=650;" style="width:416px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEB2F2DDB4A62443DF8FFE0FA660AA7B1F/4606" alt="4606" />

如果要在其他的浏览器则需要使用这个URL地址:

http://localhost:8080/TestResponse/index.html

不要直接在HBuilder中运行这个html文件,因为它的URL是指向HBuilder的工程路径的。


控制台打印结果:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEE68FE832B49C4CF0B27438E877859CD3/4607" alt="4607" />

如图,可以看到我们将所有的值都获得到手了。


获得表单数据的时候要注意一个问题:当你需要获得一个属性的值时,如果得到的结果为null,那么就是因为表单数据中并没有这个属性的存在。例如我获得一个不存在的属性:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEF402537A92144537A7299B50F7BE56E8/4608" alt="4608" />


控制台打印结果:

650) this.width=650;" style="width:258px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE2F936713C55B4C7286403D737E92F0B6/4609" alt="4609" />

可以看到结果为null,所以当你获得表单数据进行某些操作时,出现了空指针异常的话,很有可能就是因为代码上写错了获得了一个不存在的属性。


如果表单数据中的某个属性值没有写,那么获得的将是一个空字符串,而非null,例如:


650) this.width=650;" style="width:302px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE9864BF46D5754C66880529A5DBEEABA0/4610" alt="4610" />


控制台打印结果:

650) this.width=650;" style="width:441px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE7F988603D36E4D2EAF45EBC3480216D4/4611" alt="4611" />

如图,并没有打印null,而是打印空白,这个空白就是一个空字符串:’’




会出现乱码的情况,以及解决方法:

现在我们修改一下代码把表单提交的方法改为post,再运行一次,看看控制台的打印结果,html代码示例:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEB77A16689B594599A2A7027B197FCA24/4612" alt="4612" />


Java代码示例:

650) this.width=650;" style="width:552px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE1EC5BE82182A4F8C85EEFBA1CBCBCDE6/4613" alt="4613" />


提交的表单:

650) this.width=650;" style="width:337px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICED1CBB9CF74E845A2A3A38A36239BDEA0/4614" alt="4614" />


控制台的打印结果:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE8B3C4F8529B8434983F34595844B12EC/4615" alt="4615" />


可以看到控制台中的打印结果出现了不能识别的字符,解决方法很简单,使用setCharacterEncoding(String)方法,设置表单提交的数据的编码格式即可:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEC1D8E343B1034FA5B08BE0644941C61E/4616" alt="4616" />


运行结果:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE3930315358144BCF99AB68C238384DBA/4617" alt="4617" />


注意:除了在Java代码中需要设置编码格式,在html文件中也要设置好编码格式,如果html中不设置编码格式的话,即便在Java代码中使用了setCharacterEncoding(String)方法设置了也没有用,所以这是双向的,例如我把html文件中设置编码格式的标签给删掉:

650) this.width=650;" style="width:403px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEDFB533F451404D96AD8A84E19BF94E94/4618" alt="4618" />

可以看到在网页上显示都是乱码(这是因为Eclipse内置的浏览器原因,一般市面上的浏览器提前预设了字符编码,所以不会出现这种情况)


控制台打印结果:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE9803E29C5CD64098BCBA1D8076000677/4619" alt="4619" />

果然出现了不能识别的字符,所以html文件也是需要设置好编码的,不然的话就会出现乱码的情况。


下面来看看浏览器的地址栏中为什么能够显示中文:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE48DA45C6368B45FE8791F3D0F2418874/4620" alt="4620" />


这其实是因为浏览器转码了,可以把这个URL复制到记事本中:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE45265FBBFAE641089DB10CA6C93E78F8/4621" alt="4621" />

可以看到是一堆的编码,并没有显示中文,所以实际上浏览器就是把这个编码给转换成了中文而已。

只要不属于128个字符内的字符,在地址栏中都会转换成这种格式的编码,这些编码格式是采用的16进制的编码格式,以上面这文本示例编码对应的中文:

650) this.width=650;" style="width:241px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE3C7C8C039854498F8C6896ABE1A6CB05/4622" alt="4622" />

如图,每一个16进制编码都是以%开头,这是utf-8编码的中文,所以一个中文字对应3个16进制编码。

如果是GBK编码格式的中文则是一个中文字对应2个16进制编码,但是GBK编码格式转换成的16进制编码不能被浏览器转换,会仍然显示着16进制编码:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE0078016CBA3442028AC8BC3E50513287/4623" alt="4623" />


中文字对应的16进制编码:

650) this.width=650;" style="width:166px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEE6B0A7EA14DE4724AB28664DD0493097/4624" alt="4624" />

如图,GBK编码格式的中文字和utf-8编码的中文字不一样,是2个16进制编码对应一个中文字。


关于客户端请求数据方面的乱码情况就介绍这么多,另外响应数据中出现乱码的情况和解决方法在介绍HttpServletResponse方法部分进行说明。


思维导图:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE7F6DDA0A948E4C2986EC7900736CE72C/4625" alt="4625" />





HttpServletResponse中的方法

HttpServletResponse接口类型的对象是封装服务端响应数据的,所以这个对象中的方法都是与响应数据相关。以下罗列一些常用的方法:

方法名称

作用

encodeURL(String)

对给定的URL进行编码

sendError(int)

发出错误状态码

sendError(int, String)

发出错误状态码,并输出一个字符串

sendRedirect(String)

跳转页面

getOutputStream()

得到8位的输出流

getWriter()

得到16位的输出流

setBufferSize(int)

设置缓存流大小


下面使用实际的例子,演示以上方法的使用方式:

650) this.width=650;" style="width:552px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEF93F86D2FB014A04B1B83E752DA5415B/4626" alt="4626" />


编辑响应头一类的方法:

方法名称

作用

setCharacterEncoding(String)

设置响应数据的编码格式

setContentLengthLong(long)

告诉浏览器响应的数据长度

setContentType(String)

告诉浏览器本次响应的数据类型


代码示例:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEF60AFA9FFA8D4437B6BB796C403C2C4B/4627" alt="4627" />

在服务端设置响应数据的编码格式是很有必要的,这么做同样的也是为了避免出现乱码的问题。例如以下这个示例,我不设置响应数据的编码格式,并输出一段中文,看看会发生什么,代码示例:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEE235F5220C3847F796C55FA0DC378083/4628" alt="4628" />


运行结果:

650) this.width=650;" style="width:449px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE82115E73290D4D34BFA87CB690975C9C/4629" alt="4629" />

如图,可以看到,没有设置响应数据的编码格式的话,输出中文就会无法被识别。

这种问题设置一下响应数据的编码格式就好了,但是服务端设置的编码格式,要与浏览器端的编码格式对应上,如果不对应的话仍然会是乱码,代码示例:

650) this.width=650;" style="width:552px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE91E22CF1239A4C90915D3E16D853F0C8/4630" alt="4630" />


运行结果:

650) this.width=650;" style="width:397px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE87C5FB01BF164378820DBA3E1AC9A937/4631" alt="4631" />




添加新的响应头数据:

方法名称

作用

addDateHeader(String, long)

添加一个长整型的时间值

addHeader(String, String)

添加一对键/值(值的类型为字符串类型)

addIntHeader(String, int)

添加一对键/值(值的类型为整型类型)

代码示例:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICED10D79F108D54BF7878E163F5A2E2633/4632" alt="4632" />


打开TCP/IP Monitor窗口,可以看到以上代码添加进响应头的数据:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEE10164D31E8145D4AFE524FC07E0EE87/4633" alt="4633" />



获得设置的响应头信息:

方法名称

作用

getHeader(String)

参数为键,获得该键的值

getHeaderNames()

获得所有的键

getHeaders(String)

参数为键,获得拆分的值

代码示例:

650) this.width=650;" style="width:552px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE76900B9F5720421E82430655FC803EF3/4634" alt="4634" />


控制台打印结果:

650) this.width=650;" style="width:408px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICEE672E17EC47645ACACF43330D25DB400/4635" alt="4635" />



修改响应头信息:

方法名称

作用

setDateHeader(String, long)

修改一个长整型的时间值

setHeader(String, String)

修改指定的键的值(值的类型为字符串类型)

setIntHeader(String, int)

修改指定的键的值(值的类型为整型类型)

代码示例:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE06AD71721AF8473A84A46620A9877B9F/4636" alt="4636" />


TCP/IP Monitor窗口:

650) this.width=650;" style="width:553px;" src="http://note.youdao.com/yws/public/resource/7c9db10efc18662b7d0eb4cf0f9ea799/xmlnote/OFFICE1B70951B8FEE43A9861A528E540C696E/4637" alt="4637" />



总结:

  •   解决客户端表单提交数据乱码的问题,需要使用setCharacterEncoding(String)方法,设置好与客户端对应的编码格式。

  •   解决服务端响应数据乱码的问题,则使用setCharacterEncoding(String)方法,设置好对应的编码格式。

  •   HttpServletRequest是封装请求数据的对象,所以它的方法都是与客户端请求信息相关的。

  •   HttpServletResponse是封装响应数据的对象,所以它的方法都是与服务端响应信息相关的。





本文出自 “zero” 博客,请务必保留此出处http://zero01.blog.51cto.com/12831981/1980454

关于乱码问题的解决与HttpServletResponse中的方法

原文地址:http://zero01.blog.51cto.com/12831981/1980454

知识推荐

我的编程学习网——分享web前端后端开发技术知识。 垃圾信息处理邮箱 tousu563@163.com 网站地图
icp备案号 闽ICP备2023006418号-8 不良信息举报平台 互联网安全管理备案 Copyright 2023 www.wodecom.cn All Rights Reserved