1.实现第一个Filter

首先在Eclipse中创建一个名为chapter08的Web项目,然后在该项目的src目录下创建一个名为baixiao.filter包,
最后在该包下创建一个名为MyServlet的Servlet 类,该类用于访问时在浏览器中输出“Hello MyServlet”。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
package baixiao.filter;

import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;


public class MyServlet extends HttpServlet {
private static final long serialVersionUID = 1L;


protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
response.getWriter().print("hello dashuju2004-------pidan<br />");
}


protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
doGet(request, response);
}

}

在web.xml文件中对Servlet进行如下配置

1
2
3
4
5
6
7
8
9
10
<servlet>
<description></description>
<display-name>MyServlet</display-name>
<servlet-name>MyServlet</servlet-name>
<servlet-class>baixiao.filter.MyServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>MyServlet</servlet-name>
<url-pattern>/MyServlet</url-pattern>
</servlet-mapping>

启动Tomcat 服务器,在浏览器的地址栏中输入地址“http://localhost:8080/chapterO8/MyServlet”

在pidan.filter包中创建一个名为MyFilter的Fiter类,该类用于拦截MyServlet程序。

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
30
package baixiao.filter;

import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;


public class MyFilter implements Filter {

public void destroy() {

}

public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
response.setContentType("text/html;charset=utf-8");
System.out.println("拦截了原网站的响应");
response.getWriter().print("这是filter输出到网页的内容");


}

public void init(FilterConfig fConfig) throws ServletException {

}

}

过滤器程序与Servlet程序类似,同样需要在 web.xml文件中进行配置,从而设置它所能拦截的资源

1
2
3
4
5
6
7
8
<filter>
<filter-name>MyFilter</filter-name>
<filter-class>pidan.filter.MyFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>MyFilter</filter-name>
<url-pattern>/MyServlet</url-pattern>
</filter-mapping>

在上述代码中,过滤器的配置信息中包含多个元素,这些元素分别具有不同的作用,具体如下。

根元素用于注册一个Filter。
子元素用于设置Filter名称。
子元素用于设置Filter类的完整名称。
根元素用于设置一个过滤器所拦截的资源。
子元素必须与中的子元素相同。
子元素用于匹配用户请求的URL,例如“/MyServlet”,这个URL还可以使用通配符“”来表示,例如“.do”适用于所有以“.do”结尾的Servlet路径。

重新启动Tomcat 服务器,在浏览器访问MyServlet

2.Filter映射

2.1拦截不同方式的访问请求
在web.xml文件中,一个元素用于配置一个Fiter所负责拦截的资源。
元素中有一个特殊的子元素,该元素用于指定过滤器所拦截的资源被Servlet容器调用的方式,、
元素的值共有4个:

REQUEST
当用户直接访问页面时,Web容器将会调用过滤器。如果目标资源是通过RequestDispatcher的include()或 forward()方法访问的,那么该过滤器将不会被调用。
INCLUDE
如果目标资源是通过RequestDispatcher的 include()方法访问的,那么该过滤器将被调用。
除此之外,该过滤器不会被调用。
FORWARD
如果目标资源是通过RequestDispatcher的forward()方法访问的,那么该过滤器将被调用。
除此之外,该过滤器不会被调用。
ERROR
如果目标资源是通过声明式异常处理机制调用的,那么该过滤器将被调用。除此之外,过滤器不会被调用。

在chapter08项目的baixiao.filter包中,
创建一个名为ForwardServlet的Servlet类,该类用于将请求转发给first.jsp页面

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

package baixiao.filter;

import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;


public class ForwardServlet extends HttpServlet {
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
request.getRequestDispatcher("/first.jsp").forward(request,response);
}

protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
}

}

在web.xml文件中,配置ForwardServlet信息

1
2
3
4
5
6
7
8
<servlet>
<servlet-name>ForwardServlet</servlet-name>
<servlet-class>baixiao.filter.ForwardServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>ForwardServlet</servlet-name>
<url-pattern>/ForwardServlet</url-pattern>
</servlet-mapping>

在WebContent目录中创建一个first.jsp页面,该页面用于输出内容

1
2
3
4
5
6
7
8
9
10
11
12
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
first.jsp
</body>
</html>

在baixiao.filter包中,创建一个过滤器ForwardFiter.java,该过滤器专门用于对first.jsp页面进行拦截

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
package baixiao.filter;

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;

public class ForwardFilter implements Filter {

public void destroy() {

}

public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
PrintWriter out=response.getWriter();
out.write("Hello FilterYest");
}

public void init(FilterConfig fConfig) throws ServletException {

}

}

在web.xml文件中,配置过滤器的映射信息,拦截first.jsp页面

1
2
3
4
5
6
7
8
<filter>
<filter-name>ForwardFilter</filter-name>
<filter-class>baixiao.filter.ForwardFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>ForwardFilter</filter-name>
<url-pattern>/first.jsp</url-pattern>
</filter-mapping>

启动Tomcat 服务器,在浏览器访问ForwardServlet

浏览器可以正常访问JSP页面,说明ForwardFilter没有拦截到ForwardServlet转发的first.jsp页面。
为了拦截ForwardServlet通过forward()方法转发的first.jsp页面,需要在web.xml文件中的对应过滤器配置信息中增加一个子元素,将该元素的值设置为FORWARD

1
2
3
4
5
6
7
8
9
<filter>
<filter-name>ForwardFilter</filter-name>
<filter-class>baixiao.filter.ForwardFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>ForwardFilter</filter-name>
<url-pattern>/first.jsp</url-pattern>
<dispatcher>FORWARD</dispatcher>
</filter-mapping>

浏览器窗口显示的是ForwardFilter类中的内容,而first.jsp页面的输出内容没有显示。
由此可见,ForwardServlet中通过 forward()方法转发的 first.jsp页面被成功拦截了。

3.Filter链

3.1 在 chapter08 项目的pidan.filter 包中新建两个过滤器 MyFilter01 和MyFilter02

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
30
package baixiao.filter;

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;

public class MyFilter01 implements Filter {


public void destroy() {

}

public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
PrintWriter out=response.getWriter();
out.write("Hello MyFilter01<br />");
chain.doFilter(request, response);
}

public void init(FilterConfig fConfig) throws ServletException {

}

}
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
30
31
32
package baixiao.filter;

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;


public class MyFilter02 implements Filter {


public void destroy() {

}

public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
PrintWriter out=response.getWriter();
out.write("MyFilter02 Before<br />");
chain.doFilter(request, response);
out.write("MyFilter02 After<br />");
}

public void init(FilterConfig fConfig) throws ServletException {

}

}

为了防止其他过滤器影响此次Filter链的演示效果,请先在web.xml文件中注释掉其他过滤器的配置信息。
然后,将MyFilter01和MyFilter02过滤器的映射信息配置在MyServlet配置信息前面

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
<filter>
<filter-name>MyFilter01</filter-name>
<filter-class>pidan.filter.MyFilter01</filter-class>
</filter>
<filter-mapping>
<filter-name>MyFilter01</filter-name>
<url-pattern>/MyServlet</url-pattern>
</filter-mapping>
<filter>
<filter-name>MyFilter02</filter-name>
<filter-class>baixiao.filter.MyFilter02</filter-class>
</filter>
<filter-mapping>
<filter-name>MyFilter02</filter-name>
<url-pattern>/MyServlet</url-pattern>
</filter-mapping>

MyServlet首先被MyFilter01拦截了,打印出MyFilter01中的内容,然后被MyFilter02拦截,
直到MyServlet被MyFilterO2放行后,浏览器才显示出MyServlet中的输出内容。

4.FilterConfig接口

在 chapter08项目的baixiao.filter包中创建过滤器MyFilter03,来获取 web.xml中设置的参数

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
30
package baixiao.filter;

import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;


public class MyFilter03 implements Filter {
private String characterEncoding;
FilterConfig fc;

public void destroy() {

}

public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
characterEncoding=fc.getInitParameter("encoding");
System.out.println("encoding 初始化参数的值为:"+characterEncoding);
chain.doFilter(request, response);
}

public void init(FilterConfig fConfig) throws ServletException {
this.fc=fConfig;
}

}

在 web.xml文件中配置过滤器信息。由于Filter链中各个Filter的拦截顺序与它们在web.xml文件中元素的映射顺序一致,
因此,为了防止其他Filter影响MyFilter03的拦截效果,这里将MyFilter03映射信息配置在web.xml文件最前端

1
2
3
4
5
6
7
8
9
10
11
12
<filter>
<filter-name>MyFilter03</filter-name>
<filter-class>baixiao.filter.MyFilter03</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>GBK</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>MyFilter03</filter-name>
<url-pattern>/MyServlet</url-pattern>
</filter-mapping>

使用Filter获取到了配置文件中的初始化参数。当Tomcat服务器启动时,会加载所有的Web应用,
当加载到chapter08这个Web应用时,MyFiter03就会被初始化调用 init()方法,从而可以得到FilterConfig 对象。
然后在doFilter()方法中通过调用FilterConfig对象的qetlnitParameter()方法便可以获取在web.xml文件中配置的某个参数信息。

5.任务——使用Filter实现用户自动登录

5.1 编写User类
在chapter08项目中创建baixiao.entity包,在该包中编写User 类,该类用于封装用户的信息

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
package baixiao.entity;

public class User {
private String username;
private String password;
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
}

5.2 实现登录页面和首页
(1).在chapter08项目的WebContent根目录中,编写login.jsp页面,该页面用于创建一个用户登录的表单,
这个表单需要填写用户名和密码,以及用户自动登录的时间

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
30
31
32
33
34
35
36
37
<%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8" import="java.util.*"%>
<html>
<head></head>
<center>
<h3>用户登录</h3>
</center>
<body style="text-align: center;">
<form action="${pageContext.request.contextPath }/LoginServlet"
method="post">
<table border="1" width="600px" cellpadding="0" cellspacing="0"
align="center">
<tr>
<td height="30" align="center">用户名:</td>
<td>&nbsp;&nbsp; <input type="text" name="username" />${errorMsg }</td>
</tr>
<tr>
<td height="30" align="center">密 &nbsp; 码:</td>
<td>&nbsp;&nbsp; <input type="password" name="password" /></td>
</tr>
<tr>
<td height="35" align="center">自动登录时间</td>
<td><input type="radio" name="autologin"
value="${60*60*24*31 }" />一个月 <input type="radio" name="autologin"
value="${60*60*24*31*3 }" />三个月 <input type="radio"
name="autologin" value="${60*60*24*31*6 }" />半年 <input
type="radio" name="autologin" value="${60*60*24*31*12 }" />一年</td>
</tr>
<tr>
<td height="30" colspan="2" align="center"><input type="submit"
value="登录" /> &nbsp;&nbsp;&nbsp;&nbsp; <input type="reset"
value="重置" /></td>
</tr>
</table>
</form>
</body>
<html>

(2).在chapter08项目的WebContent根目录中,编写index.jsp页面,该页面用于显示用户的登录信息。
如果没有用户登录,在index.jsp页面中就显示一个用户登录的超链接。如果用户已经登录,在 index.isp页面中显示登录的用户名,以及一个注销的超链接

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
  <%@ page language="java" contentType="text/html; charset=utf-8"
pageEncoding="utf-8" import="java.util.*"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<html>
<head>
<title>显示登录的用户信息</title>
</head>
<body>
<br />
<center>
<h3>欢迎光临</h3>
</center>
<br />
<c:choose>
<c:when test="${sessionScope.user==null }">
<a href="http://localhost:9999/chap08/login.jsp">用户登录</a>
</c:when>
<c:otherwise>
欢迎您:${sessionScope.user.username} <a
href="http://localhost:9999/chap08/LogoutServlet">退出</a>
</c:otherwise>
</c:choose>
<hr />
</body>
</html>

6.创建Servlet
(1)编写LoginServlet类
在chapter08项目的baixiaon.filter包中,编写LoginServlet类,该类用于处理用户的登录请求。
如果输入的用户名和密码正确,则发送一个用户自动登录的Cookie,并跳转到首页;否则会提示输入的用户名或密码错误,
并跳转至登录页面login.jsp让用户重新登录

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
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
package baixiao.filter;

import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import pidan.entity.User;

public class LoginServlet extends HttpServlet {

public LoginServlet() {
super();
}

protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
String username = request.getParameter("username");
String password = request.getParameter("password");
if ("pidan".equals(username) && "666666".equals(password)) {
User user = new User();
user.setUsername(username);
user.setPassword(password);
request.getSession().setAttribute("user", user);
String autoLogin = request.getParameter("autoLogin");
if (autoLogin != null) {
Cookie cookie = new Cookie(autoLogin, username + "-"+password);
cookie.setMaxAge(Integer.parseInt(autoLogin));
cookie.setPath(request.getContextPath());
response.addCookie(cookie);
}
response.sendRedirect(request.getContextPath()+"/index.jsp");
}else {
request.setAttribute("errerMsg", "用户名或密码错误");
request.getRequestDispatcher("/login.jsp").forward(request, response);
}
}

protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
doGet(request, response);
}

}

7.创建过滤器
在chapter08项目的baixiao.filter包中,编写AutoLoginFilter类,该类用于拦截用户登录的访问请求,
判断请求中是否包含用户自动登录的Cookie。
如果包含,则获取Cookie中的用户名和密码,并验证用户名和密码是否正确。如果正确,则将用户的登录信息封装到User对象存入Session域中,完成用户自动登录

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
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
package baixiao.filter;

import java.io.IOException;
import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;

import baixiao.entity.User;

public class AutoLoginFilter implements Filter {

public void destroy() {

}

public void doFilter(ServletRequest req, ServletResponse response, FilterChain chain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) req;
Cookie[] cookies = request.getCookies();
String autologin = null;
for (int i = 0; cookies != null && i<cookies.length;i++) {
if ("autologin".equals(cookies[i].getValue())) {
autologin = cookies[i].getValue();
break;
}
}
if (autologin != null) {
String[] parts = autologin.split("-");
String username = parts[0];
String password = parts[1];
if ("pidan".equals(username)&&("666666").equals(password)) {
User user = new User();
user.setUsername(username);
user.setPassword(password);
request.getSession().setAttribute("user", user);
}
}
chain.doFilter(request, response);
}

public void init(FilterConfig fConfig) throws ServletException {
}

}

8.配置映射信息
在web.xml文件中,配置所有相关Servlet及AutoLoginFilter过滤器信息。
由于要拦截用户访问资源的所有请求,因此,将过滤器元素拦截的路径设置为“/”

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
<filter>
<filter-name>AutoLoginFilter</filter-name>
<filter-class>baixiao.filter.AutoLoginFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>AutoLoginFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<servlet>
<servlet-name>LoginServlet</servlet-name>
<servlet-class>baixiao.filter.LoginServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>LoginServlet</servlet-name>
<url-pattern>/LoginServlet</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>LogoutServlet</servlet-name>
<servlet-class>baixiao.filter.LogouServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>LogoutServlet</servlet-name>
<url-pattern>/LogoutServlet</url-pattern>
</servlet-mapping>

9.运行项目,查看结果
(1)访问login.jsp页面
重启服务器,打开IE浏览器在地址栏中输入“http://localhost:8888/chapter08/login.jsp”,
此时,浏览器窗口中会显示一个用户登录的表单,在这个表单中输入用户名“baixiao”、密码“123456”,并选择用户自动登录的时间
(2)实现登录
点击登录按钮,便可完成自动登录
(3)注销用户
单击图中的【注销】超链接,就可以注销当前的用户,然后显示index.jsp页面