Servle第四篇(会话技术之cookie)

会话技术

什么是会话技术

   基本概念: 指用户开一个浏览器,访问一个网站,只要不关闭该浏览器,不管该用户点击多少个超链接,访问多少资源,直到用户关闭浏览器,整个这个过程我们称为一次会话.
为什么我们要使用会话技术?

  •   在论坛登陆的时候,很多时候会有一个小框框问你是否要自动登陆,当你下次登陆的时候就不用输入密码了
  •   根据我以前浏览过的商品,猜我喜欢什么商品

Cookie

会话跟踪技术有Cookie和Session,Cookie技术是先出现的。我们先讲Cookie技术吧。
什么是Cookie
  Cookie是由W3C组织提出,最早由netscape社区发展的一种机制
  网页之间的交互是通过HTTP协议传输数据的,而Http协议是无状态的协议。无状态的协议是什么意思呢?一旦数据提交完后,浏览器和服务器的连接就会关闭,再次交互的时候需要重新建立新的连接。
  服务器无法确认用户的信息,于是乎,W3C就提出了:给每一个用户都发一个通行证,无论谁访问的时候都需要携带通行证,这样服务器就可以从通行证上确认用户的信息。通行证就是Cookie
Cookie的流程:浏览器访问服务器,如果服务器需要记录该用户的状态,就使用response向浏览器发送一个Cookie,浏览器会把Cookie保存起来。当浏览器再次访问服务器的时候,浏览器会把请求的网址连同Cookie一同交给服务器。
Cookie API
  Cookie类用于创建一个Cookie对象
  response接口中定义了一个addCookie方法,它用于在其响应头中增加一个相应的Set-Cookie头字段
  request接口中定义了一个getCookies方法,它用于获取客户端提交的Cookie
  常用的Cookie方法:
    public Cookie(String name,String value)
    setValue与getValue方法
    setMaxAge与getMaxAge方法
    setPath与getPath方法
    setDomain与getDomain方法
    getName方法
简单使用Cookie
  创建Cookie对象,发送Cookie给浏览器、
  记得设置有效期setMaxAge()

response.setContentType("text/html;charset=UTF-8");
Cookie cookie=new Cookie("username", "yolo");
cookie.setMaxAge(1000);
response.addCookie(cookie);
response.getWriter().write("我已向浏览器发送cookie");


Cookie细节
  Cookie不可跨域名性
    浏览器判断一个网站是否能操作另一个网站的Cookie的依据是域名。所以一般来说,当我访问baidu的时候,浏览器只会把baidu颁发的Cookie带过去,而不会带上google的Cookie。即使是同一级域名,不同二级域名也不能交接
  Cookie保存中文
    中文属于Unicode字符,英文数据ASCII字符,中文占4个字符或者3个字符,英文占2个字符。
    解决:Cookie使用Unicode字符时需要对Unicode字符进行编码---------Cookie cookie=new Cookie("username", URLEncoder.encode(name,"UTF-8"));
    取出Cookie的时候要对中文数据进行解码--------
  Cookie的有效期
    如果MaxAge为正数,浏览器会把Cookie写到硬盘中,只要还在MaxAge秒之前,登陆网站时该Cookie就有效【不论关闭了浏览器还是电脑】
    如果MaxAge为负数,Cookie是临时性的,仅在本浏览器内有效,关闭浏览器Cookie就失效了,Cookie不会写到硬盘中。Cookie默认值就是-1。这也就为什么在我第一个例子中,如果我没设置Cookie的有效期,在硬盘中就找不到对应的文件。
    如果MaxAge为0,则表示删除该Cookie。Cookie机制没有提供删除Cookie对应的方法,把MaxAge设置为0等同于删除Cookie
  Cookie的修改和删除
    Cookie的名称相同,通过response添加到浏览器中,会覆盖原来的Cookie。
    删除时也要addCookie
    注意:删除,修改Cookie时,新建的Cookie除了value、maxAge之外的所有属性都要与原Cookie相同。否则浏览器将视为不同的Cookie,不予覆盖,导致删除修改失败!
  Cookie的域名
    Cookie的domain属性决定运行访问Cookie的域名。domain的值规定为“.域名”

    希望同一级域名可以访问------cookie.setDomain(".zhongfucheng.com");
  Cookie的路径
    Cookie的path属性决定允许访问Cookie的路径--------使用Servlet2颁发一个Cookie给浏览器,设置路径为"/Servlet1"。
  Cookie的安全属性
    HTTP协议不仅仅是无状态的,而且是不安全的!如果不希望Cookie在非安全协议中传输,可以设置Cookie的secure属性为true,浏览器只会在HTTPS和SSL等安全协议中传输该Cookie。
    当然了,设置secure属性不会将Cookie的内容加密。如果想要保证安全,最好使用md5算法加密
Cookie的应用
  显示用户上次访问的时间
    其实就是每次登陆的时候,取到Cookie保存的值,再更新下Cookie的值。

     SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        response.setContentType("text/html;charset=UTF-8");
        PrintWriter printWriter=response.getWriter();
        
        Cookie[] cookies=request.getCookies();
        
        String cookieValue=null;
        for(int i=0;cookies!=null&&i<cookies.length;i++){
            if(cookies[i].getName().equals("time")){
                printWriter.write("您上次登录的时间是");
                cookieValue=cookies[i].getValue();
                printWriter.write(cookieValue);
                
                cookies[i].setValue(simpleDateFormat.format(new Date()));
                response.addCookie(cookies[i]);
                
                break;
            }
        }

        if(cookies==null){
            Cookie cookie=new Cookie("time",simpleDateFormat.format(new Date()));
            cookie.setMaxAge(2000);
            response.addCookie(cookie);
            printWriter.write("您是第一次登陆");
        }
    }

  显示上次浏览过商品

1、新建一个Book类

public class Book {

    private String id;
    private String name;
    private String author;
    
    public Book(){
        
    }
    
    public Book(String id,String name,String author){
        this.id=id;
        this.name=name;
        this.author=author;
    }
  各种get,setXxx

  2、数据库类DB

public class DB {
    private static LinkedHashMap<String, Book> linkedHashMap=new LinkedHashMap<String, Book>();
    static{
        linkedHashMap.put("1", new Book("1", "javaWeb", "y"));
        linkedHashMap.put("2", new Book("2", "Servlet", "o"));
        linkedHashMap.put("3", new Book("3", "JSP", "l"));
        linkedHashMap.put("4", new Book("4", "SQL", "陈"));
    }
    public static LinkedHashMap<String, Book> getAll(){
        return linkedHashMap;
}
}

  3、首页的代码(读取数据库显示,读取cookies)

    //实现展示书籍并配超链接
        response.setContentType("text/html;charset=UTF-8");
        PrintWriter printWriter=response.getWriter();
        printWriter.write("网页上所有的书籍"+"<br/>");
        LinkedHashMap<String,Book> linkedHashMap=DB.getAll();
        Set<Map.Entry<String, Book>> entry=linkedHashMap.entrySet();
        
        for(Map.Entry<String, Book> stringBookEntry:entry){
            Book book=stringBookEntry.getValue();
            printWriter.write("<a href='/firstServlet/Demo?id="+book.getId()+"''target=_blank'>"+book.getName()+"</a>");
            printWriter.write("<br/>");
        }
        
        //显示历史记录
        printWriter.write("您浏览过的商品有");
        printWriter.write("<br/>");
        
        Cookie[] cookies=request.getCookies();
        for(int i=0;cookies!=null&&i<cookies.length;i++){
            if(cookies[i].getName().equals("bookHistory")){
                String bookHistory=cookies[i].getValue();
             
                String[] ids=bookHistory.split("\\_");
                
                for(String id:ids){
                    Book book=linkedHashMap.get(id);
                    printWriter.write(book.getName());
                    printWriter.write("<br/>");
                }
                break;
            }
        }

  4、超链接跳转后处理数据

private String makeHistory(HttpServletRequest request,String id){
        String bookHistory=null;
        Cookie[] cookies=request.getCookies();
      //找到匹配的cookies
for(int i=0;cookies!=null&&i<cookies.length;i++){ if(cookies[i].getName().equals("bookHistory")){ bookHistory=cookies[i].getValue(); } } if(bookHistory==null){ return id; } String[] strings=bookHistory.split("\\_"); //为了方便增删减改 List<String> list=Arrays.asList(strings); LinkedList<String> linkedList=new LinkedList<String>(); linkedList.addAll(list); //排序 if(linkedList.contains(id)){ linkedList.remove(id); linkedList.addFirst(id); }else if(linkedList.size()>=3){ linkedList.removeLast(); linkedList.addFirst(id); }else{ linkedList.addFirst(id); } //拼接 StringBuffer stringBuffer=new StringBuffer(); for(String s:linkedList){ stringBuffer.append(s+"_"); } return stringBuffer.deleteCharAt(stringBuffer.length()-1).toString(); }

  5、超链接页面显示以及将处理后的数据存到cookie

     response.setContentType("text/html;charset=UTF-8");
        String id=request.getParameter("id");
        Book book=(Book)DB.getAll().get(id);
        
        PrintWriter printWriter=response.getWriter();
        printWriter.write("编号"+book.getId()+"<br/>");
        printWriter.write("名称"+book.getName()+"<br/>");
        printWriter.write("作者"+book.getAuthor()+"<br/>");
            
        String bookHistory=makeHistory(request,id);
        Cookie cookie=new Cookie("bookHistory", bookHistory);
        cookie.setMaxAge(20000);
        response.addCookie(cookie);

 

posted on 2019-03-18 09:43  诗悦网络内推有问必答  阅读(165)  评论(0编辑  收藏  举报

导航