在上一篇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类删掉呦。
谢谢观看