1.过滤器的引入:
场景1: 编写servlet,接收参数:
request.getParameter() /getParameterValues()
如果不做任何处理,就出现中文乱码问题。
解决编码问题: request.setCharacterEncoding("utf-8");
问题:如果在项目中的每一个servlet都加上request.setCharacterEncoding("utf-8");
显示代码重复啰嗦。能不能把这部分公共代码抽取处理,放在一个地方执行????
场景2: 登录 -> 输入信息 -> 登录成功 -> 看到用户主页(欢迎xxx回来。。。)
用于验证用户是否登录成功代码:
if(session==null){
跳转到登录页面
}else{
loginName = session.getAttribute("loginName");
if(loginName==null){
跳转到登录页面
}
}
-> 个人信息修改页面
-> 个人密码修改页面
如果用户不登录,直接访问用户主页,跳转到登录页面
在其他需要登录才能访问的页面中,同样也需要加上验证用户是否登录成功代码。
问题: 能不能把这部分公共验证用户是否登录成功代码抽取处理,在一个地方执行??
结论: 以上两种场景出现的问题,可以使用过滤器(Filter)解决!!!!
2.过滤器的接口
1)过滤器其实就是一个接口,Filter, javax.servet.Filter
2)过滤器就是一个对象,可以在请求一个资源(静态或动态资源),或响应一个资源,或请求和响应一个资源的时候,执行过滤任务!!!!
3)过滤器如何被执行?过滤器也需要交给tomcat服务器运行!!!!
Servlet的三大组件:( 1)都需要交给web服务器运行 2)在web.xml文件中配置 )
Servlet接口
Filter接口
Listener接口
4)过滤器的生命周期
构造方法: 在web应用加载时创建过滤器对象。只执行一次。证明过滤器在web服务器中是单实例的
init方法: 在创建完过滤器对象之后被调用。只执行一次
doFilter方法: 执行过滤任务方法。执行多次。
destroy方法: web服务器停止或者web应用重新加载,销毁过滤器对象。
5)过滤器编写步骤:
5.1 编写一个java类,实现Filter接口,并实现其中的所有方法
5.1 在web.xml文件中配置Filter
<!-- 过滤器配置 -->
<filter>
<!-- 内部名称 -->
<filter-name>HelloFilter</filter-name>
<!-- 全名:包+简单类名 -->
<filter-class>gz.itcast.a_hello.HelloFilter</filter-class>
</filter>
<!-- 过滤器映射配置 -->
<filter-mapping>
<!-- 内部名称,和上面的名称保持一致! -->
<filter-name>HelloFilter</filter-name>
<!-- 需要拦截的路径 -->
<url-pattern>/hello</url-pattern>
</filter-mapping>
5.3 把Filter部署到tomcat服务器运行!!!!
3.过滤器的映射路径
过滤器中的url-pattern: 表示的这个过滤器需要拦截的目标资源路径(可以servlet路径,也可以是静态资源名称)
Servlet中的url-pattern: 表示访问这个servlet时的路径 url-pattern 浏览器访问目标资源的路径
精确过滤 /hello http://localhost:8080/day21/hello
/itcast/hello http://localhost:8080/day21/itcast/hello
模糊过滤 /* http://localhost:8080/day21/任意路径
/itcast/* http://localhost:8080/day21/itcast/任意路径
*.后缀名 http://localhost:8080/day21/任意路径.后缀名
注意:
1)url-pattern要么以斜杠开头,要么以*开头 例如: hello
2)不能同时使用两个模糊过滤。例如 /*.do 是非法的
3)如果存在多个需要被过滤的资源,可以写多个url-pattern去过滤。
4)如果是动态资源servlet,可以使用servlet的访问名称,也可以使用内部名称
<!-- 使用servlet的内部名称 -->
<servlet-name>HelloServlet</servlet-name>
5)过滤类型:
<!-- 过滤类型:声明哪种请求才可以被拦截(过滤) -->
<dispatcher>REQUEST</dispatcher><!-- 默认:来自于直接访问的请求才可以被拦截 -->
<dispatcher>FORWARD</dispatcher><!-- 来自于转发的请求才可以被拦截 -->
<dispatcher>INCLUDE</dispatcher><!-- 来自于包含的请求才可以被拦截 -->
<dispatcher>ERROR</dispatcher><!-- 来自于错误的请求才可以被拦截 -->
4.FilterConfig对象
FilterConfig对象,过滤器配置对象,用于加载过滤器的参数配置
过滤器参数使用:
1)在web.xml文件中配置
<!-- 过滤器配置 -->
<filter>
<!-- 内部名称 -->
<filter-name>HelloFilter</filter-name>
<!-- 类全名:包+简单类名 -->
<filter-class>gz.itcast.a_hello.HelloFilter</filter-class>
<init-param>
<param-name>AAA</param-name>
<param-value>AAA'value</param-value>
</init-param>
<init-param>
<param-name>BBB</param-name>
<param-value>BBB'value</param-value>
</init-param>
</filter>
2)在过滤器器中使用
/ * 2)init初始化方法
*/
public void init(FilterConfig filterConfig) throws ServletException {
System.out.println("2)Filter生命周期-init方法");
/**
* 通过FilterConfig对象得到参数配置信息
*/
//得到一个参数
System.out.println(filterConfig.getInitParameter("AAA"));
Enumeration<String> enums = filterConfig.getInitParameterNames();
//遍历所有参数
while(enums.hasMoreElements()){
String paramName = enums.nextElement();
String paramValue = filterConfig.getInitParameter(paramName);
System.out.println(paramName+"="+paramValue);
}
}
5.过滤器链
doFilter(ServletRequest request, ServletResponse response, FilterChain chain): 执行过滤任务
参数一: ServletRequest是HttpServletRequest的父接口。实际上传入的是HttpServletRequest接口的实现类。
参数二: ServletResponse是HttpServletResponse的父接口。实际上传入HttpServletResponse接口的实现类。
参数三: FilterChain 过滤器链接口
doFilter(ServletRequest request, ServletResponse response):执行过滤器链中的下一个过滤器,如果没有下一个过滤器则执行目标资源。
*****过滤器链: 一个目标资源可以被多个过滤器过滤,那么形成一个过滤器链。***
注意:过滤器链中的过滤器执行顺序问题:由web.xml中filter-mapping的配置决定顺序。先配置的优先被执行
6.装饰者模式
者模式(Decorator)
23种java设计模式。单例模式,工厂模式,适配器模式,观察者模式,代理模式。。。。。。
装饰者模式:当开发者觉得某些类的某些方法不满足需要,向增强这些类的方法。这是就可以使用装饰者模式去装饰这些类。不满足需求的这些类,叫被装饰类。开发者需要重新编写装饰类去覆盖被装饰类。
BufferedReader:被装饰类。readLine方法不满足需要。
MyBufferReader: 装饰类。
装饰步骤:
1)继承被装饰类。BufferedReader。非final
2)装饰类中声明一个成员变量(被装饰类类型)
3)在构造方法,把被装饰类的实例接收到,赋值给成员变量
4)重写被装饰的方法。
7.案列(过滤器的使用)
6.1 使用过滤器完美解决GET和Post方式提交中文乱码问题
6.2 使用过滤器解决网页内容压缩问题
为什么要进行网页内容压缩?
访问web服务器时,服务器会返回网页内容(数据)
用户访问一个页面:
100KB
100万个用户访问这个页面
1 ,000,000 * 100Kb = 100,000,000 = 服务器消耗了100G内容 (带宽)
买服务器:
网络服务器运营商
按流量收费: 我们尽可能压缩网页内容,才输出给浏览器
怎么对网页内容压缩?
可以是java 的GZIPOutputStream类进行gzip压缩。