Servlet(Server Applet),全称Java Servlet,是用Java编写的服务器端程序。其主要功能在于交互式地浏览和修改数据,生成动态Web内容。本系列将一步步地写出一个Servlet程序。
这篇博文将演示如何使用cookie
和session
进行会话追踪。
HTTP协议是一个无状态的协议,也就是说,在服务器眼中,每一个HTTP请求都是一个全新的请求,每个请求之间没有关联。所以我们需要一个可以管理请求中携带的用户信息的方法。而会话追踪就是一个可以管理用户信息的方法。
会话追踪可以通过下列几个方式实现:
- Cookie
- 表单隐藏域
- URL改写
- HttpSession
本文将主要演示Cookie
和HttpSession
的用法。
Cookie
什么是cookie
Cookie
是一串可以持久化于各个请求之间的信息片段。每个cookie都有一个名字,并有一个值,同时可以包含备注、路径、域名、过期时间、版本等附加信息。
Cookie有两种:
- 非持久cookie,这种cookie只在会话中存留,并且不具有过期时间属性,一旦用户关闭浏览器(或者标签页),也就是使这个会话失效,这个cookie就会丢失。
- 持久化cookie,这种cookie可以被用于多个会话中,而且只会在到达过期时间,或者用户主动使该cookie失效后,才会被删除。
可以使用HttpServletResponse#addCookie(Cookie)
方法在HTTP响应中携带cookie。
保存cookie
首先修改前文中的doPost()
方法,将请求中的参数取出来,并存入cookie。
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
| @Override public void doPost(HttpServletRequest request, HttpServletResponse response) throws IOException { response.setContentType("text/html");
response.setCharacterEncoding("utf-8");
Map parameterMap = request.getParameterMap();
try (PrintWriter writer = response.getWriter()) { writer.print("<html lang=\"en\">"); writer.print("<body>"); writer.print("<b>Response from DemoServlet</b>"); writer.print("<br>"); writer.print("<b>Handled by <code>doPost()</code></b>"); writer.print("<br>");
writer.print("<b>Session ID: " + request.getSession().getId() + "</b>"); writer.print("<br>");
parameterMap.forEach((k, v) -> { writer.print(k + " = " + ((String[]) v)[0] + "<br>");
response.addCookie(new Cookie(String.valueOf(k), ((String[]) v)[0])); });
writer.print("</body>"); writer.print("</html>"); } }
|
然后发送一个POST
请求,在返回中可以看到请求中的参数已经被放到cookie中,并返回到了客户端。
data:image/s3,"s3://crabby-images/7057c/7057cc3cfbe67c8cb2a9311eb0b909b98ebf4181" alt="POST request with cookie"
使用cookie
一旦cookie被保存到了客户端,那么在下次访问这个cookie所对应的地址时,客户端就会自动将相关的cookie带入请求一并发送到服务端。所以客户端不需要对cookie主动做任何操作。
修改前文中的doGet()
方法,使其可以取出cookie的值,并输出到页面上。
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 50 51 52 53 54 55 56 57 58
| @Override public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException { response.setContentType("text/html");
response.setCharacterEncoding("utf-8");
Optional<String> optionalQueryString = Optional.ofNullable(request.getQueryString());
String[] queryStrings = optionalQueryString.isPresent() ? optionalQueryString.get().split("&") : new String[]{};
try (PrintWriter writer = response.getWriter()) { writer.print("<html lang=\"en\">"); writer.print("<body>"); writer.print("<b>Response from DemoServlet</b>"); writer.print("<br>"); writer.print("<b>Handled by <code>doGet()</code></b>"); writer.print("<br>");
writer.print("<b>Session ID: " + request.getSession().getId() + "</b>"); writer.print("<br>");
writer.print("<br>");
writer.print("<b>Parameters: </b>"); writer.print("<br>");
for (String query : queryStrings) { String[] q = query.split("=");
writer.print(q[0] + " = " + q[1] + "<br>"); }
writer.print("<br>"); writer.print("<b>Cookies:</b>"); writer.print("<br>");
Cookie[] cookies = request.getCookies();
for (Cookie cookie : cookies) { writer.print(cookie.getName() + " = " + cookie.getValue()); writer.print("<br>"); }
writer.print("</body>"); writer.print("</html>"); } }
|
然后发送一个GET
请求,在返回中可以看到cookie中的内容已经被输出到页面上。
data:image/s3,"s3://crabby-images/06744/0674473605cb7fe5e15c4c7022a83f1c62c87565" alt="GET request with cookie"
删除cookie
将cookie的存活时间设为0,并返回到客户端,即可从客户端中删除这个cookie。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| @Override protected void doDelete(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { Cookie[] cookies = req.getCookies();
for (Cookie cookie : cookies) { if (!"JSESSIONID".equalsIgnoreCase(cookie.getName())) { cookie.setMaxAge(0);
resp.addCookie(cookie); } }
try (PrintWriter writer = resp.getWriter()) { writer.print(""); } }
|
HttpSession
什么是session
Session记录着一次会话相关的信息。
当一个请求到达服务器后,服务器会检查请求中是否包含session ID信息,比如在Tomcat中就是检查有无JSESSIONID
这个cookie,或者URL中有无JSESSIONID
这个查询字符串。如果找到了对应的session,则服务器会将这个session检索出来使用;请求中没有包含session ID,或者对应的session已经被销毁,则服务器会创建一个新的session并返回其ID。
Session ID通常以cookie的形式返回到客户端,如果客户端禁用了cookie,那么服务端则会使用URL重写技术将session ID写到URL中。
Session中可以键值对的形式保存附加数据,称为attributes。
与cookie不同,session保存于服务器端,而且它能保存的数据也不仅限于字符串。
保存attribute
修改doPost()
方法,编写修改session的代码。修改完成后发送一个带有参数的POST
请求,以向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
| @Override public void doPost(HttpServletRequest request, HttpServletResponse response) throws IOException { response.setContentType("text/html");
response.setCharacterEncoding("utf-8");
Map parameterMap = request.getParameterMap();
HttpSession session = request.getSession();
try (PrintWriter writer = response.getWriter()) { writer.print("<html lang=\"en\">"); writer.print("<body>"); writer.print("<b>Response from DemoServlet</b>"); writer.print("<br>"); writer.print("<b>Handled by <code>doPost()</code></b>"); writer.print("<br>");
writer.print("<b>Session ID: " + request.getSession().getId() + "</b>"); writer.print("<br>");
parameterMap.forEach((k, v) -> { writer.print(k + " = " + ((String[]) v)[0] + "<br>");
Cookie cookie = new Cookie(String.valueOf(k), ((String[]) v)[0]); response.addCookie(cookie);
session.setAttribute(String.valueOf(k), ((String[]) v)[0]); });
writer.print("</body>"); writer.print("</html>"); } }
|
取出attribute
修改doGet()
方法,使其可以从session中取出attributes并显示在页面上。
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 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
| @Override public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException { response.setContentType("text/html");
response.setCharacterEncoding("utf-8");
Optional<String> optionalQueryString = Optional.ofNullable(request.getQueryString());
HttpSession session = request.getSession();
Enumeration<String> attributeNames = session.getAttributeNames();
String[] queryStrings = optionalQueryString.isPresent() ? optionalQueryString.get().split("&") : new String[]{};
try (PrintWriter writer = response.getWriter()) { writer.print("<html lang=\"en\">"); writer.print("<body>"); writer.print("<b>Response from DemoServlet</b>"); writer.print("<br>"); writer.print("<b>Handled by <code>doGet()</code></b>"); writer.print("<br>");
writer.print("<b>Session ID: " + request.getSession().getId() + "</b>"); writer.print("<br>");
writer.print("<br>");
writer.print("<b>Parameters: </b>"); writer.print("<br>");
for (String query : queryStrings) { String[] q = query.split("=");
writer.print(q[0] + " = " + q[1] + "<br>"); }
writer.print("<br>"); writer.print("<b>Cookies:</b>"); writer.print("<br>");
Cookie[] cookies = request.getCookies();
for (Cookie cookie : cookies) { writer.print(cookie.getName() + " = " + cookie.getValue()); writer.print("<br>"); }
writer.print("<br>"); writer.print("<b>Attributes: </b>"); writer.print("<br>");
while(attributeNames.hasMoreElements()) { String key = attributeNames.nextElement();
String value = String.valueOf(session.getAttribute(key));
writer.print(key + " = " + value); writer.print("<br>"); }
writer.print("</body>"); writer.print("</html>"); } }
|
然后发送一个GET
请求,在返回中就可以看到刚才保存在session中的数据:
data:image/s3,"s3://crabby-images/336ce/336ceb53d96b3a7b7d0441724e79a4d2413e22a2" alt="GET request with session attribute"
删除attribute
此外HttpSession
类提供了removeAttribute()
方法用于删除一个attribute。
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
| @Override protected void doDelete(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { Cookie[] cookies = req.getCookies();
HttpSession session = req.getSession();
Enumeration<String> attributeNames = session.getAttributeNames();
for (Cookie cookie : cookies) { if (!"JSESSIONID".equalsIgnoreCase(cookie.getName())) { cookie.setMaxAge(0);
resp.addCookie(cookie); }
while(attributeNames.hasMoreElements()) { String key = attributeNames.nextElement();
session.removeAttribute(key); } }
try (PrintWriter writer = resp.getWriter()) { writer.print(""); } }
|
系列博文