博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
会话管理之Cookie技术
阅读量:5960 次
发布时间:2019-06-19

本文共 8142 字,大约阅读时间需要 27 分钟。

目录

        会话管理是web开发中比较重要的环节,这一节主要总结下会话管理中的cookie技术。

1. 何为会话

        会话可简单理解为:用户开一个浏览器,点击多个超链接,访问服务器多个web资源,然后关闭浏览器,整个过程称为一个会话。

1.1 会话过程中要解决的问题

        1)每个用户在使用浏览器与服务器进行会话的过程中,不可避免会各自产生一些数据,程序要想办法为每个用户保存这些数据。

        2)如:用户点击超链接通过一个servlet购买了一个商品,程序应该想办法保存用户购买的商品,以便于用于点结账servlet时,结账servlet可以得到用户购买的商品为用户结账。
        3)思考:用户购买的商品保存在request或servletContext中行不行?
        如果保存在request中,假设用户购买一台洗衣机,首先访问buy的servlet,然后将洗衣机保存到该servlet的request中,接下来用户要结账,又访问pay的servlet,该servlet会拿到一个全新的request,并没有刚刚保存过的洗衣机。如果保存在servletContext中,假如用户购买一台洗衣机,首先访问buy的servlet,然后将洗衣机存入servletContext中,用户结账访问pay的servlet,可以拿到刚刚的洗衣机。但是有个问题,在没结账之前,另一个浏览器也在买东西,且买了电视机,也会存到servletContext中,这将覆盖掉刚刚的洗衣机,刚刚买的洗衣机就不见了。

        这就需要我们使用其他技术去保存会话过程中产生的数据了。

1.2 保存会话数据的两种技术

        保存会话数据有两种技术:cookie技术和session技术。

        cookie是客户端技术,程序把每个用户的数据以cookie的形式写给用户各自的浏览器,当用户使用浏览器再去访问服务器中的web资源时,就会带着各自的数据去。这样web资源处理的就是用户各自的数据了。

        session是服务器端技术,利用这个技术,服务器在运行时可以为每一个用户的浏览器创建一个其独享的session对象,由于session为用户浏览器独享,所以用户在访问服务器的web资源时,可以把各自的数据放在各自的session中,当用户再去访问服务器中的其它web资源时,其它web资源再从用户各自的session中取出数据位用户服务。

这一节我们主要总结下cookie技术的使用。

2. cookie的相关API

        javax.servlet.http.Cookie类用于创建一个cookie,response接口也定义了一个addCookie方法,它用于在其响应头中中增加一个相应的set-Cookie头字段。同样,request接口中也定义了一个getCookies方法,它用于获取客户端提交的Cookie。Cookie类的方法有:

[java]   
 
  1. public Cookie(String name, String value);  //new一个cookie的时候需要指定cookie的名称和值  
  2. setValue() getValue();  //设置和获得cookie的值  
  3. setMaxAge() getMaxAge();  //设置和获得cookie的有效期  
  4. setPath() getPath();  //设置和获得带cookie的访问路径  
  5. setDomain() getDomain(); //设置和获得带cookie的访问域  
  6. getName();  //获得cookie的名称  
  7. //注:括号为空并不表示没有参数,这里只是表示一下方法名而已   
        下面通过一个很简单的示例程序来看一下cookie API的使用:

[java]   
 
  1. //显示用户上次访问的时间  
  2. public class CookieDemo1 extends HttpServlet {  
  3.     public void doGet(HttpServletRequest request, HttpServletResponse response)  
  4.             throws ServletException, IOException {        
  5.         response.setContentType("text/html; charset=UTF-8");//设置编码格式  
  6.           
  7.         PrintWriter out = response.getWriter();  
  8.         out.write("这是网站首页<br/><br/>");  
  9.         out.write("您上次访问网站的时间:");  
  10.           
  11.         //得到上次访问的时间  
  12.         Cookie cookies[] = request.getCookies();//获得浏览器带过来的所有cookie  
  13.         for(int i = 0; cookies != null && i < cookies.length; i++) {  
  14.             Cookie cookie = cookies[i];   //遍历每个cookie  
  15.             if(cookie.getName().equals("lastAccessTime")) {  //通过cookie的名字判断是不是想要的cookie  
  16.                 Long time = Long.parseLong(cookie.getValue());  //如果是,拿到该cookie的值  
  17.                 Date d = new Date(time);   //转换成时间的格式  
  18.                 out.write(d.toLocaleString());  //打印到浏览器  
  19.             }  
  20.         }  
  21.           
  22.         //如果没拿到cookie,说明第一次访问,那么给用户以cookie的形式送最新的时间  
  23.         Cookie cookie = new Cookie("lastAccessTime", System.currentTimeMillis() + ""); //cookie的名字可以自己指定  
  24.         cookie.setMaxAge(10000);//cookie保存10000秒  
  25.         response.addCookie(cookie);//将新建的cookie添加进去,带给浏览器  
  26.                   
  27.     }  
  28.     public void doPost(HttpServletRequest request, HttpServletResponse response)  
  29.             throws ServletException, IOException {  
  30.   
  31.         doGet(request, response);  
  32.     }  
  33.   
  34. }  
        这样我们通过访问:http://localhost:8080/test/servlet/CookieDemo1就可以看到浏览器显示上次登录的时间了。注:web.xml中需要对这个servlet配置一下,这里就不再赘述了。

3. cookie细节

        一个cookie只能标识一种信息,它至少含有一个标识该信息的名称(name)和设置值((value)。

        一个web站点可以给一个web浏览器发送多个cookie,一个web浏览器也可以存储多个web站点提供的cookie。
        浏览器一般只允许存放300个cookie,每个站点最多存放20个cookie,每个cookie的大小限制为4KB。
        如果创建一个cookie并将它发送到浏览器,默认情况下它是一个会话级别的cookie(即存储在浏览器内存里),用户退出浏览器之后即被删除。若希望浏览器将该cookie存储在磁盘上,则需要使用maxAge,并给出一个以秒为单位的时间。将最大的时间值设置为0则是命令浏览器删除该cookie。删除cookie时,path必须一致,否则不会删除。

4. cookie的应用

        这里用cookie技术模拟一个显示用户浏览过的商品。有两个servlet,一个用于显示所有商品和浏览过的商品,另一个用于显示商品具体信息并返回cookie

CookieDemo2是用来显示所有商品和浏览过的商品:

[java]   
 
  1. public class CookieDemo2 extends HttpServlet {  
  2.   
  3.     public void doGet(HttpServletRequest request, HttpServletResponse response)  
  4.             throws ServletException, IOException {  
  5.           
  6.         response.setContentType("text/html;charset=UTF-8");  
  7.         PrintWriter out = response.getWriter();       
  8.     //1. 显示网站所有商品         
  9.         out.write("本网站有如下书籍:<br>");  
  10.         //entrySet方法会返回一个set集合,set集合中保存网站所有书,保存的是Map.Entry  
  11.         Set<Map.Entry<String, Book>> set = DB.getAll().entrySet();  
  12.         for(Map.Entry<String, Book> me:set) {  
  13.             Book book = me.getValue();  
  14.             out.write("<a href='/test/servlet/CookieDemo3?id="+book.getId()+"' target='_blank'>"+book.getName()+"</a>");  
  15.             out.write("<br/>");  
  16.         }         
  17.         out.write("<br/>");  
  18.           
  19.     //2. 显示用户曾经浏览过的商品         
  20.         out.write("您曾经浏览过的商品:<br/>");  
  21.         Cookie cookies[] = request.getCookies();//获得cookie  
  22.         for(int i = 0; cookies != null && i < cookies.length; i++) {  
  23.             Cookie cookie = cookies[i];  
  24.             if(cookie.getName().equals("bookHistory")) { //找到浏览器带过来的cookie  
  25.                 String bookHistory = cookie.getValue();  
  26.                 String ids[] = bookHistory.split("\\_");//以下划线进行分隔  
  27.                 for(String id : ids){  
  28.                     Book book = (Book)DB.getAll().get(id);  
  29.                     out.write(book.getName() + "<br/>");  
  30.                 }  
  31.             }  
  32.         }         
  33.     }  
  34.   
  35.     public void doPost(HttpServletRequest request, HttpServletResponse response)  
  36.             throws ServletException, IOException {  
  37.   
  38.         doGet(request, response);  
  39.     }  
  40.   
  41. }  
  42.   
  43. //写一个类来模拟数据库  
  44. class DB {  
  45.       
  46.     private static Map<String,Book> map = new LinkedHashMap<String, Book>();  
  47.       
  48.     //静态代码块中的内容只执行一次,该类在加载时,往map集合中put一系列书,map也需要设置为静态的  
  49.     static{  
  50.           
  51.         map.put("1"new Book("1""javaweb开发","老张""一本好书"));  
  52.         map.put("2"new Book("2""spring开发","老倪""一本好书"));  
  53.         map.put("3"new Book("3""hibernate开发","老童""一本好书"));  
  54.         map.put("4"new Book("4""struts开发","老毕""一本好书"));  
  55.         map.put("5"new Book("5""ajax开发","老张""一本好书"));  
  56.         map.put("6"new Book("6""java基础","老孙""一本好书"));  
  57.           
  58.     }  
  59.       
  60.     public static Map getAll() {  
  61.         return map;  
  62.     }  
  63.       
  64. }  
  65.   
  66. class Book {      
  67.     private String id;  
  68.     private String name;  
  69.     private String author;  
  70.     private String description;  
  71.       
  72.     public Book() {  
  73.     }  
  74.   
  75.     public Book(String id, String name, String author, String description) {  
  76.         super();  
  77.         this.id = id;  
  78.         this.name = name;  
  79.         this.author = author;  
  80.         this.description = description;  
  81.     }  
  82.       
  83.     public String getId() {  
  84.         return id;  
  85.     }  
  86.     public void setId(String id) {  
  87.         this.id = id;  
  88.     }  
  89.     public String getName() {  
  90.         return name;  
  91.     }  
  92.     public void setName(String name) {  
  93.         this.name = name;  
  94.     }  
  95.     public String getAuthor() {  
  96.         return author;  
  97.     }  
  98.     public void setAuthor(String author) {  
  99.         this.author = author;  
  100.     }  
  101.     public String getDescription() {  
  102.         return description;  
  103.     }  
  104.     public void setDescription(String description) {  
  105.         this.description = description;  
  106.     }     
  107. }  
CookieDemo3用来显示商品的具体信息并返回cookie:

[java]   
 
  1. public class CookieDemo3 extends HttpServlet {  
  2.   
  3.     public void doGet(HttpServletRequest request, HttpServletResponse response)  
  4.             throws ServletException, IOException {  
  5.           
  6.         response.setContentType("text/html;charset=UTF-8");  
  7.         PrintWriter out = response.getWriter();  
  8.   
  9.     //1. 根据用户带过来的id号,显示商品的详细信息  
  10.         String id = request.getParameter("id");  
  11.         Book book = (Book)DB.getAll().get(id);  
  12.           
  13.         out.write("您要查看的书的详细信息如下:<br/>");  
  14.         out.write(book.getId() + "<br/>");  
  15.         out.write(book.getName() + "<br/>");  
  16.         out.write(book.getAuthor() + "<br/>");  
  17.         out.write(book.getDescription() + "<br/>");  
  18.           
  19.     //2. 给用户回送包含商品id的cookie  
  20.         String bookHistory = makeHistory(request, id); //生成cookie中保存的浏览历史数据(书的id)  
  21.         Cookie cookie = new Cookie("bookHistory", bookHistory);  
  22.         cookie.setMaxAge(1*30*24*3600);//cookie保存一个月  
  23.         response.addCookie(cookie);  
  24.     }  
  25.   
  26.     //用来获得一个字符串,该字符串作为名为bookHistory的cookie的值  
  27.     private String makeHistory(HttpServletRequest request, String id) {  
  28.   
  29.         String bookHistory = null;  
  30.         Cookie cookies[] = request.getCookies();  
  31.         for(int i = 0; cookies != null && i < cookies.length; i++){  
  32.             if(cookies[i].getName().equals("bookHistory")){  
  33.                 bookHistory = cookies[i].getValue();  
  34.             }  
  35.         }  
  36.           
  37.         //bookHistory=null     1    bookHistory=1  
  38.         //bookHistory=3_1_5    1    bookHistory=1_3_5  
  39.         //bookHistory=3_2_5    1    bookHistory=1_3_2  
  40.         //bookHistory=3_2      1    bookHistory=1_3_2  
  41.           
  42.         if(bookHistory == null) {  //bookHistory=null     1    bookHistory=1  
  43.             return id;  
  44.         }  
  45.           
  46.         List<String> lst = Arrays.asList(bookHistory.split("\\_"));  
  47.         LinkedList<String> list = new LinkedList<String>();  
  48.         list.addAll(lst);  
  49.           
  50.         if(list.contains(id)){  //bookHistory=3_1_5    1    bookHistory=1_3_5  
  51.             list.remove(id);  
  52.             list.addFirst(id);  
  53.         }   
  54.         else {  
  55.             if(list.size() >= 3) {  //bookHistory=3_2_5    1    bookHistory=1_3_2  
  56.                 list.removeLast();  
  57.                 list.addFirst(id);  
  58.             }  
  59.               
  60.             else {  //bookHistory=3_2      1    bookHistory=1_3_2  
  61.                 list.addFirst(id);  
  62.             }  
  63.         }  
  64.           
  65.         StringBuffer sb = new StringBuffer();  
  66.         for(String lid : list) {  
  67.             sb.append(lid + "_");  
  68.         }     
  69.         //整个for循环结束后会多了个"_",所以下面先把这个"_"删掉  
  70.         return sb.deleteCharAt(sb.length()-1).toString();  
  71.     }  
  72.   
  73.     public void doPost(HttpServletRequest request, HttpServletResponse response)  
  74.             throws ServletException, IOException {  
  75.   
  76.         doGet(request, response);  
  77.     }  
  78. }  
       以上就是一个简单的cookie小案例,其实cookie并不难,掌握了其原理和流程,就真的像是cookie单词本身的意思一样了……

       cookie就介绍这么多吧,如有错误之处,欢迎留言指正~

       相关阅读:

_____________________________________________________________________________________________________________________________________________________

-----乐于分享,共同进步!

-----更多文章请看:
你可能感兴趣的文章
hibernate简单入门教程(四)---------关联映射
查看>>
去 IOE,MySQL 完胜 PostgreSQL
查看>>
++i 和 i++ 性能上的区别
查看>>
Mysql运维管理-一主多从宕机从库切换主库继续和从库同步过程16
查看>>
Tomcat优化之配置NIO运行模式
查看>>
用XSLT和XML改进Struts
查看>>
WEB测试—功能测试
查看>>
在react或vue中,for循环用Index作为key值是好还是坏呢?
查看>>
2014.10.1 Form中显示pdf文件
查看>>
NERDTree 快捷键辑录
查看>>
Python数据分析Numpy库方法简介(一)
查看>>
javaWeb:相关监听方法汇总
查看>>
JSP 实现 之 读取数据库显示图片
查看>>
JS——特效秀
查看>>
Beta冲刺——day6
查看>>
前端:CheckBox事件函数js
查看>>
Comet OJ - Contest #3 题解
查看>>
[网络流24题-9]试题库问题
查看>>
jquery选择器详解
查看>>
C# 保留2位小数
查看>>