黑马程序员技术交流社区

标题: 关于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