目录
1过滤器入门
1.1 为什么要用过滤器
1.2 过滤器简介
2 过滤器的映射路径
3 FilterConfig对象
4 过滤器链
5 案例
5.1 使用过滤器完美解决GET和Post方式提交中文乱码问题
5.2 使用过滤器解决网页内容压缩问题
5.3 使用过滤器过滤访问权限
1过滤器入门
1.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)解决!!!!
1.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
5.3 把Filter部署到tomcat服务器运行!!!!
<!-- 过滤器配置 --> <filter> <!-- 内部名称 --> <filter-name>HelloFilter</filter-name> <!-- 类全名:包+简单类名 --> <filter-class>gz.ittest.a_hello.HelloFilter</filter-class> </filter> <!-- 过滤器映射配置 --> <filter-mapping> <!-- 内部名称,和上面的名称保持一致! --> <filter-name>HelloFilter</filter-name> <!-- 需要拦截的路径 --> <url-pattern>/hello</url-pattern> </filter-mapping> |
2 过滤器的映射路径
过滤器中的url-pattern: 表示的这个过滤器需要拦截的目标资源路径(可以servlet路径,也可以是静态资源名称)
Servlet中的url-pattern: 表示访问这个servlet时的路径
url-pattern 浏览器访问目标资源的路径
精确过滤 /hello http://localhost:8080/day21/hello
/ittest/hello http://localhost:8080/day21/ittest/hello
模糊过滤 /* http://localhost:8080/day21/任意路径
/ittest/* http://localhost:8080/day21/ittest/任意路径
*.后缀名 http://localhost:8080/day21/任意路径.后缀名
注意:
1)url-pattern要么以斜杠开头,要么以*开头 例如: hello
2)不能同时使用两个模糊过滤。例如 /*.do 是非法的
3)如果存在多个需要被过滤的资源,可以写多个url-pattern去过滤。
4)如果是动态资源servlet,可以使用servlet的访问名称,也可以使用内部名称
<!-- 使用servlet的内部名称 --> |
5)过滤类型:
<!-- 过滤类型:声明哪种请求才可以被拦截(过滤) --> <dispatcher>REQUEST</dispatcher><!-- 默认:来自于直接访问的请求才可以被拦截 --> <dispatcher>FORWARD</dispatcher><!-- 来自于转发的请求才可以被拦截 --> <dispatcher>INCLUDE</dispatcher><!-- 来自于包含的请求才可以被拦截 --> <dispatcher>ERROR</dispatcher><!-- 来自于错误的请求才可以被拦截 --> |
3 FilterConfig对象
FilterConfig对象,过滤器配置对象,用于加载过滤器的参数配置
过滤器参数使用:
1)在web.xml文件中配置
<!-- 过滤器配置 --> <filter> <!-- 内部名称 --> <filter-name>HelloFilter</filter-name> <!-- 类全名:包+简单类名 --> <filter-class>gz.ittest.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); }
} |
4 过滤器链
doFilter
(ServletRequest request, ServletResponse response, FilterChain chain)
: 执行过滤任务
参数一: ServletRequest是HttpServletRequest的父接口。实际上传入的是HttpServletRequest接口的实现类。
参数二: ServletResponse是HttpServletResponse的父接口。实际上传入HttpServletResponse接口的实现类。
参数三: FilterChain 过滤器链接口
doFilter(ServletRequest request, ServletResponse response):执行过滤器链中的下一个过滤器,如果没有下一个过滤器则执行目标资源。
*****过滤器链: 一个目标资源可以被多个过滤器过滤,那么形成一个过滤器链。***
注意:过滤器链中的过滤器执行顺序问题:由web.xml中filter-mapping的配置决定顺序。先配置的优先被执行。
5 案例
5.1 使用过滤器完美解决GET和Post方式提交中文乱码问题
5.2 使用过滤器解决网页内容压缩问题
为什么要进行网页内容压缩?
访问web服务器时,服务器会返回网页内容(数据)
用户访问一个页面: 100KB
100万个用户访问这个页面
1 ,000,000 * 100Kb = 100,000,000 = 服务器消耗了100G内容 (带宽)
用户访问一个页面: 100B
100万个用户访问这个页面
1 ,000,000 * 100B = 100,000,000 = 100M服务器消耗了100M内容 (带宽)
买服务器:
网络服务器运营商
按流量收费: 我们尽可能压缩网页内容,才输出给浏览器
怎么对网页内容压缩?
可以是java 的GZIPOutputStream类进行gzip压缩。
/** * 对网页内容进行压缩 */ //创建临时的字节数组容器 ByteArrayOutputStream byteArr = new ByteArrayOutputStream(); //创建GZIPOutputStream对象 GZIPOutputStream gzip = new GZIPOutputStream(byteArr); //开始写出压缩内容 gzip.write(sb.toString().getBytes()); //刷新缓冲区 gzip.finish();
//从临时的字节数组容器中得到压缩后的网页内容 byte[] result = byteArr.toByteArray();
System.out.println("压缩后的数据大小:"+result.length);
/** * 注意:告诉浏览器数据压缩格式 发送响应头:content-encoding:gzip */ response.setHeader("content-encoding", "gzip");
//把压缩后的内容输出到浏览器 response.getOutputStream().write(result); |