黑马程序员技术交流社区
标题:
关于web请求参数乱码问题!过滤器的!
[打印本页]
作者:
521123270
时间:
2015-6-11 11:15
标题:
关于web请求参数乱码问题!过滤器的!
过滤器解决乱码时,写一个一类继承HttpServletWrapp... 将request对象进行装饰,覆盖那个三个获取请求参数方法!请说下原理!特别是HttpServletWrapp... 它又一个父类 ,父类实现了被装饰的 request的同一接口!!关系不是很懂!求详解!{:3_65:}
作者:
三月_麦田
时间:
2015-6-11 11:15
Servlet的框架是由两个Java包组成:javax.servlet和javax.servlet.http. 在javax.servlet包中定义了所有的Servlet类都必须实现或扩展的的通用接口和类.在javax.servlet.http包中定义了采用HTTP通信协议的HttpServlet类.
Servlet的框架的核心是javax.servlet.Servlet接口,所有的Servlet都必须实现这一接口.在Servlet接口中定义了5个方法,其中有3个方法代表了Servlet的声明周期:
init方法,负责初始化Servlet对象
service方法,负责相应客户的请求
destory方法,当Servlet对象退出声明周期时,负责释放占有的资源
当Web容器接收到某个Servlet请求时,Servlet把请求封装成一个HttpServletRequest对象,然后把对象传给Servlet的对应的服务方法.
HTTP的请求方式包括DELETE,GET,OPTIONS,POST,PUT和TRACE,在HttpServlet类中分别提供了相应的服务方法,它们是,doDelete(),doGet(),doOptions(),doPost(), doPut()和doTrace().
HttpServlet的功能
HttpServlet首先必须读取Http请求的内容。Servlet容器负责创建HttpServlet对象,并把Http请求直接封装到HttpServlet对象中,大大简化了HttpServlet解析请求数据的工作量。HttpServlet容器响应Web客户请求流程如下:
1)Web客户向Servlet容器发出Http请求;
2)Servlet容器解析Web客户的Http请求;
3)Servlet容器创建一个HttpRequest对象,在这个对象中封装Http请求信息;
4)Servlet容器创建一个HttpResponse对象;
5)Servlet容器调用HttpServlet的service方法,把HttpRequest和HttpResponse对象作为service方法的参数传给HttpServlet对象;
6)HttpServlet调用HttpRequest的有关方法,获取HTTP请求信息;
7)HttpServlet调用HttpResponse的有关方法,生成响应数据;
8)Servlet容器把HttpServlet的响应结果传给Web客户。
二、创建HttpServlet的步骤——“四部曲”
1)扩展HttpServlet抽象类;
2)覆盖HttpServlet的部分方法,如覆盖doGet()或doPost()方法;
3)获取HTTP请求信息。通过HttpServletRequest对象来检索HTML表单所提交的数据或URL上的查询字符串;
4)生成HTTP响应结果。通过HttpServletResponse对象生成响应结果,它有一个getWriter()方法,该方法返回一个PrintWriter对象。
举个例子如下:
package mypack;
import javax.servlet.*;
import javax.servlet.http.*;
import java.io.*;
public class HelloServlet extends HttpServlet//第一步:扩展HttpServlet抽象类
{
//第二步:覆盖doGet()方法
public void doGet(HttpServletRequest request,
HttpServletResponse response)throws IOException,ServletException{
//第三步:获取HTTP请求中的参数信息
String clientName=request.getParameter("clientName");
if(clientName!=null)
clientName=new String(clientName.getBytes("ISO-8859-1"),"GB2312");
else
clientName="我的朋友";
//第四步:生成HTTP响应结果
PrintWriter out;
String title="HelloServlet";
String heading1="HelloServlet的doGet方法的输出:";
//set content type
response.setContentType("text/html;charset=GB2312");
//write html page
out=response.getWriter();
out.print("<HTML><HEAD><TITLE>"+title+"</TITLE>");
out.print("</HEAD><BODY>");
out.print(heading1);
out.println("<h1><p>"+clientName+":您好</h1>");
out.print("</BODY></HTML>");
out.close();
}
}
在web.xml中添加
<servlet>
<servlet-name>HelloServlet</servlet-name>
<servlet-class>mypack.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>HelloServlet</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
通过URL访问HelloServlet:
注意:
实现service方法。
Servlet的主要功能是接受从浏览器发送过来的HTTP请求(request),并返回HTTP响应(response)。这个工作是在service方法中完成的。service方法包括从request对象获得客户端数据和向response对象创建输出。
如果一个Servlet从javax.servlet.http.HttpServlet继承,实现了doPost或doGet方法,那么这个Servlet只能对POST或GET做出响应。如果开发人员想处理所有类型的请求(request),只要简单地实现service方法即可(但假如选择实现service方法,则不必实现doPost或doGet方法,除非在service方法的开始调用super.service())。
作者:
三月_麦田
时间:
2015-6-16 15:57
Servlet的框架是由两个Java包组成:javax.servlet和javax.servlet.http. 在javax.servlet包中定义了所有的Servlet类都必须实现或扩展的的通用接口和类.在javax.servlet.http包中定义了采用HTTP通信协议的HttpServlet类.
作者:
逝....曾经
时间:
2015-6-18 12:54
Servlet的框架是由两个Java包组成:javax.servlet和javax.servlet.http. 在javax.servlet包中定义了所有的Servlet类都必须实现或扩展的的通用接口和类.在javax.servlet.http包中定义了采用HTTP通信协议的HttpServlet类
作者:
黑马无敌
时间:
2015-6-23 15:16
获取数据时,服务器端的每个程序都需要针对POST,GET进行设置,且都是同样的写法,那就显得太不优雅了,所以最后用一个过滤器替换掉对所有页面的request和response的设置.那我就需要在过滤器里面对request或者response对象进行增强,然后把增强后的request或者response放行,我在服务器程序里面就不用再写解决乱码的代码,从而提高程序的可维护性.
增强一个类有三种方式:
直接继承该类
包装设计模式
动态代理
显然,这里不适合用继承,虽然它最简单,但是一个request对象中包含着众多的数据,如果想创造一个request对象,就必须知道request是如何产生的,然后用服务器创造request对象的方式去创造我的"MyRequest"对象,来达到增强的目的.所以在这里放弃这种方式.
过滤器+包装类
public class CharacterFilter implements Filter {
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) req;
HttpServletResponse response = (HttpServletResponse) res;
//POST的乱码解决方案
request.setCharacterEncoding("utf-8");
//返回数据的乱码解决方案
response.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
//将增强后的对象放行
MyRequest myRequest = new MyRequest(request);
chain.doFilter(myRequest, response); //这样一来,后面所有的操作都是基于这个增强后的对象进行的
}
/**创建一个request对象的包装类:
1.编写一个类,实现与被增强对象相同的接口
2.在类中定义一个变量,记住被增强对象
3.在类中定义一个构造方法,接收被增强对象
4.覆写想要增强的方法
5.对于不想增强的方法,直接调用被增强对象(目标对象)的方法
包装设计模式"五步曲"
*/
class MyRequest extends HttpServletRequestWrapper{
private HttpServletRequest request;
public MyRequest(HttpServletRequest request) {
super(request);
this.request = request;
}
@Override
public String getParameter(String name) {
//如果请求方式是POST,则不用增强,直接调用目标对象的方法
if(this.request.getMethod().equalsIgnoreCase("POST")){
return this.request.getParameter(name);
}
/*程序运行到此,请求方式必然为GET
先获取值,再进行手动转换*/
String value = this.request.getParameter(name);
try {
value = new String(value.getBytes("iso8859-1"),"utf-8");
} catch (UnsupportedEncodingException e) {
throw new RuntimeException(e);
}
return value; //返回
}
}
public void destroy() {
// TODO Auto-generated method stub
}
public void init(FilterConfig fConfig) throws ServletException {
// TODO Auto-generated method stub
}
}
过滤器记得在web.xml中配置一把
过滤器+动态代理
public class CharacterFilter2 implements Filter {
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException {
final HttpServletRequest request = (HttpServletRequest) req;
HttpServletResponse response = (HttpServletResponse) res;
//POST的乱码解决方案
request.setCharacterEncoding("utf-8");
//返回数据的乱码解决方案
response.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=utf-8");
//用动态代理拦截,增强getParameter()后,放行
chain.doFilter((ServletRequest) Proxy.newProxyInstance(CharacterFilter2.class.getClassLoader(), request.getClass().getInterfaces(),
new InvocationHandler(){ //直接实现接口
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
//如果请求方式是POST,则不用增强,直接调用目标对象的方法
if(request.getMethod().equalsIgnoreCase("POST")){
return method.invoke(request, args);
}
String methodName = method.getName();
//如果传递进来的方法不是getParameter(),则不用增强
if(!methodName.equals("getParameter")){
return method.invoke(request, args);
}
//为GET,并且是getParameter(),...
String value = (String) method.invoke(request, args);
if(value!=null){
value = new String(value.getBytes("iso8859-1"),"utf-8");
}
return value;
}
}), response);
}
public void destroy() {
// TODO Auto-generated method stub
}
public void init(FilterConfig fConfig) throws ServletException {
// TODO Auto-generated method stub
}
}
作者:
zhangshuai_123
时间:
2015-7-5 11:10
你只要
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.http.HttpServletRequest;public class PageEncodeFilter implements Filter { private FilterConfig fConfig;
private String encode;
public void destroy() {
this.fConfig = null;
this.encode = null;
} public void doFilter(ServletRequest arg0, ServletResponse arg1,
FilterChain fc) throws IOException, ServletException {
HttpServletRequest hsr = (HttpServletRequest)arg0;
hsr.setCharacterEncoding(this.encode);
fc.doFilter(arg0, arg1);
} public void init(FilterConfig fConfig) throws ServletException {
this.fConfig = fConfig;
this.encode = this.fConfig.getInitParameter("encode");
}}然后到web.xml 下配置 <filter>
<filter-name>PageEncodeFilter</filter-name>
<filter-class>com.popedom.util.PageEncodeFilter</filter-class>
<init-param>
<param-name>encode</param-name>
<param-value>gb2312</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>PageEncodeFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
作者:
q953655369
时间:
2015-7-5 21:28
不野不知道
作者:
小绿
时间:
2015-7-8 23:21
http://s1429908977.blog.163.com/blog/static/24983307220155141112
欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/)
黑马程序员IT技术论坛 X3.2