如果你想开发一个应用(1-7)

在上一篇blog里,将todo列表实现了不同用户列表的隔离功能,但是想一想,代码中是否存在着很多不和谐的地方?

不和谐的地方其实很多,但这一章先解决一个最简单的地方,注意一下TodoServlet的post方法的第一行:

request.setCharacterEncoding("utf-8");

他的作用在第二章的时候就已经介绍过来,是为了解决中文乱码的问题,但是,难道我们每个页面,每个servlet的各种方法中都要写么?假如有个企业应用,有100个jsp页面怎么办呢?

方法当然有,这里可以使用servlet api的过滤器类来实现

Filter

具体实现方式,首先还是先创建包:

com.niufennan.jtodos.filter

然后创建类EncodingFilter使其集成Filter接口.

public class EncodingFilter implements Filter {
       public void init(FilterConfig filterConfig) throws ServletException {
    }

    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
    }

    public void destroy() {
    }
}

此接口必须实现三个方法,分别为初始化操作,执行期操作,销毁期操作,因为我们在这个过滤器创建即销毁时均没有特殊操作,所以只要完成doFilter方法即可:

public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
    servletRequest.setCharacterEncoding("utf-8");
    filterChain.doFilter(servletRequest,servletResponse);
}

最后,为了减少(不使用)xml配置,使用注解来配置过滤器名和过滤路径:

@WebFilter( filterName = "encodingFilter",urlPatterns="/*")

filterName代表过滤器的唯一性名字,urlPatterns代码过滤的路径,可以使用作为通配符,其中/\表示过滤所有路径。

删除TodoServlet类的代码request.setCharacterEncoding("utf-8");后,运行输入中文测试无乱码效果。

过滤器执行生命周期

修改filter类,在三个生命周期方法中输出字符:

public void init(FilterConfig filterConfig) throws ServletException {
    System.out.println("filter-init");
}

public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
    servletRequest.setCharacterEncoding("utf-8");
    System.out.println("filter-doFilter");
    filterChain.doFilter(servletRequest,servletResponse);
}

public void destroy() {
    System.out.println("filter-destroy");
}

启动测试服务器,并仔细观察控制台,发现init的输出实在Artifact is deployed successfully的时候,即容器加载这个过滤器完成之后立即进行初始化。在之后,根据你访问的页面不同,调用了若干次的doFilter,但这时候发现,并没有调用destory方法。

这时候,停止测试服务器,你很快就发现,对destory进行了调用,即对过滤器进行了销毁。

Servlet类的生命周期与过滤器一致。

过滤链

这时候,你也许会问了,如果多个过滤器怎么办?这个,来做个测试吧,修改这个测试的输出字符:

public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
    servletRequest.setCharacterEncoding("utf-8");
    System.out.println("Bef_EncodingFilter");
    filterChain.doFilter(servletRequest,servletResponse);
    System.out.println("Aft_EncodingFilter");
}

并新增一个测试,同样输出字符,这样就可以很清晰了:

@WebFilter(filterName = "TestFilter",urlPatterns = "/*")
public class TestFilter implements Filter {
    public void init(FilterConfig filterConfig) throws ServletException {
    }
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        System.out.println("Bef_TestFilter");
        filterChain.doFilter(servletRequest,servletResponse);
        System.out.println("Aft_TestFilter");
    }
    public void destroy() {
    }
}

同时,在Servlet中也增加输出语句:

System.out.println("TodoServlet");

运行后,输出顺序为:

Bef_EncodingFilter
Bef_TestFilter
TodoServlet
Aft_TestFilter
Aft_EncodingFilter

执行顺序一目了然:

请原谅我的灵魂画风

执行顺序

那么,最后一个问题,为什么先执行EncodingFilter这个过滤器呢,难道是因为它比较早创建么?

答案当然是否定的,修改一下TestFilter这个类的名字:

@WebFilter(filterName = "AestFilter",urlPatterns = "/*")
public class AestFilter implements Filter {
    public void init(FilterConfig filterConfig) throws ServletException {
    }
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        System.out.println("Bef_TestFilter");
        filterChain.doFilter(servletRequest,servletResponse);
        System.out.println("Aft_TestFilter");
    }
    public void destroy() {
    }
}

再次执行,查看输出:

Bef_TestFilter
Bef_EncodingFilter
TodoServlet
Aft_EncodingFilter
Aft_TestFilter

顺序就发生了变化,很奇怪Tomcat这个servelt容器使用类名,或者说文件名来决定加载顺序,这一点不如通过xml配置的方式,可以显式的决定过滤器的加载顺序。

最后,别忘了把测试的Filter类删掉呦。

谢谢观看