JavaWeb三大组件
1. 都需要在web.xml中进行配置
Servlet
Listener
Filter
2. 过滤器
它通过web.xml管理着一大片资源,它会在一组资源(jsp、servlet、.css、.html等等)的前面执行!它当你想要访问它管理的资源时,那么它就会拦截进行处理.它可以让请求得到目标资源,也可以不让请求达到!就好比如门卫. 只要是对很多的资源进行操作就应该想到filter
过滤器如何编写
1. 写一个类实现Filter接口
2. 在web.xml中进行配置管理哪些资源
1. Filter接口
1 2 3 4 5 6 |
|
void init(FilterConfig)
* 创建之后,马上执行;Filter会在服务器启动时就创建!
void destory()
* 销毁之前执行!在服务器关闭时销毁
void doFilter(ServletRequest,ServletResponse,FilterChain)
* 每次过滤时都会执行
Filter是单例的!和servlet一样
2. web.xml
<filter>
<filter-name>xxx</filter-name>
<filter-class>web.filter.AFitler</fitler-class>
</servlet>
<fitler-mapping>
<filter-name>xxx</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
这个配置代表了这个filter管理哪些资源.访问这些资源时,这个过滤器就会执行doFilter()..
FilterConfig-->与ServletConfig相似
* 获取初始化参数:getInitParameter()
* 获取过滤器名称:getFilterName()
* 获取appliction:getServletContext()
FilterChain(这个用于放行)
* doFilter(ServletRequest, ServletResponse):放行!
就相当于调用了目标Servlet的service()方法!
-------------------------------
-------------------------------
多过滤器情况下FilterChain的doFilter()方法:执行目标资源,或是执行下一个过滤器!如果没有下一个过滤器那么执行的是目标资源,如果有,那么就执行下一个过滤器!
-------------------------------
过滤器的四种拦截方式
<dispatcher>REQUEST</dispatcher>默认的!拦截直接请求
<dispatcher>FORWARD</dispatcher> 拦截请求转发
<dispatcher>INCLUDE</dispatcher> 拦截请求包含
<dispatcher>ERROR</dispatcher> 拦截错误转发
在<filter-mapping>中进行配置!
-------------------------------
多个过滤器的执行顺序
<filter-mapping>的配置顺序决定了过滤器的执行顺序!
过滤器的应用场景:
执行目标资源之前做预处理工作,例如设置编码,这种试通常都会放行,只是在目标资源执行之前做一些准备工作;
通过条件判断是否放行,很多java教程里都这样说的,例如校验当前用户是否已经登录,或者用户IP是否已经被禁用;
在目标资源执行后,做一些后续的特殊处理工作,例如把目标资源输出的数据进行处理
实例:
统计IP
循环遍历在ServletContext中的map,其中key是ip地址,value是访问次数
jsp页面
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
|
filter代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 |
|
解决全站字符乱码
乱码问题:
l 获取请求参数中的乱码问题;
POST请求:request.setCharacterEncoding(“utf-8”);
GET请求:newString(request.getParameter(“xxx”).getBytes(“iso-8859-1”), “utf-8”);
l 响应的乱码问题:response.setContextType(“text/html;charset=utf-8”)。
处理POST请求简单,可是处理get请求需要获取参数,filter获取所有过滤资源的参数是不可能实现的.这里可以调包request再发给servlet.增强request对象(改变getParamater()方法.让每次获取参数时直接解决乱码问题).
增强对象的方式有三种:(对a对象进行增强,fun1()方法)
- 继承: AA类继承a对象的类型:A类,然后重写fun1()方法,其中重写的fun1()方法就是被增强的方法。但是,继承必须要知道a对象的真实类型,然后才能去继承。如果我们不知道a对象的确切类型,而只知道a对象是IA接口的实现类对象,那么就无法使用继承来增强a对象了;
- 装饰者模式: AA类去实现a对象相同的接口:IA接口,还需要给AA类传递a对象,AA类所有方法的实现都是调用a对象相同方法实现,只有fun1()方法需要改变下内容,对fun1()进行增强;
- 动态代理:和增强者模式比较相似
这里对request对象进行增强是通过继承request对象的装饰类,装饰类是接口的包装类,但是它不进行任何增强,我们通过继承它然后重写需要增强的方法,这样就不用重写需要增强的方法了.
(增强时一看继承类,二看有没有接口的包装类,三接口的装饰者模式)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
|
继承包装类增强request
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
|