Cookie和Session

本文深入解析Cookie和Session的工作原理及应用,对比两者的区别,包括客户端和服务端技术的区别,以及如何在实际项目中使用它们进行会话管理和数据共享。

文字简单解释Cookie和Sessio传输的不同

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

保存会话数据的两种技术:Cookie,Session

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

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

Cookie应用

//显示用户上次访问的时间
public class CookieDemo1 extends HttpServlet {

    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        response.setCharacterEncoding("UTF-8");
        response.setContentType("text/html;charset=UTF-8");

        PrintWriter  out = response.getWriter();
        out.write("您上次访问时间是:");
        //1.获取用户上次访问的时间,显示
        Cookie cookies[] = request.getCookies();
        for(int i=0;cookies!=null && i<cookies.length;i++){
            Cookie cookie = cookies[i];
            if(cookie.getName().equals("lastAccessTime")){
                long time = Long.parseLong(cookie.getValue());
                Date date = new Date(time);
                out.write(date.toLocaleString());
            }
        }

        //2.把本次的时间以cookie的形式回写给客户机   (lastAccessTime)
        Cookie cookie = new Cookie("lastAccessTime",System.currentTimeMillis()+"");
        cookie.setMaxAge(100);//设置有效期,cookie将写到硬盘上,不设置有效期,cookie的有效期是会话范围,cookie仅存储在浏览器缓存中,浏览器一关闭,cookie就没了。
        response.addCookie(cookie);

    }

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

}

Cookie细节:

  1. 一个Cookie只能标识一种信息,它至少含有一个标识该信息的名称(NAME)和设置值(VALUE)。
  2. 一个WEB站点可以给一个WEB浏览器发送多个Cookie,一个WEB浏览器也可以存储多个WEB站点提供的Cookie。
  3. 浏览器一般只允许存放300个Cookie,每个站点最多存放20个Cookie,每个Cookie的大小限制为4KB。
  4. 如果创建了一个cookie,并将他发送到浏览器,默认情况下它是一个会话级别的cookie(即存储在浏览器的内存中),用户退出浏览器之后即被删除。若希望浏览器将该cookie存储在磁盘上,则需要使用maxAge,并给出一个以秒为单位的时间。将最大时效设为0则是命令浏览器删除该cookie。
  5. 注意,删除cookie时,path必须一致,否则不会删除

session应用

1.设置Session

String data = "abcd";
HttpSession session = request.getSession();
session.setAttribute("data", data);

2.取出Session

HttpSession  session = request.getSession();
//session.invalidate();  摧毁session

String data = (String) session.getAttribute("data");
response.getWriter().write(data);

3.session实现会话过程数据共享的原理:

Session是基于Cookie的、服务器创建每个session的时候,会为每个session分配一个ID号,即 JSESSIONID,服务器会以cookie的方式把session 的id号写给浏览器、所以session是基于cookie的。当浏览器再次访问的时候,会带着session id号的cookie过来 ,这就是session实现共享的原理。服务器写这个cookie的时候,是没有写有效期的,所以当浏览器关闭的时候,cookie也就没了,session也就结束了,即会话过程。

4.多浏览器共享Session

由上面的分析可以,关闭浏览器后,session将消失,那么如何做到多浏览器共享一个session呢(即关闭浏览器后,再打开一个浏览器后session还存在)?

只要手动的写入cookie,并设置有效期即可。用这个cookie来覆盖服务器写给浏览器的。可以通过httpwatch来观察服务器写给浏览器的内容。

解决方案如下:名字必须是JSESSIONID,因为服务器写给浏览器的cookie的名字就是这个 。

HttpSession session = request.getSession();
        Cookie cookie = new Cookie("JSESSIONID",session.getId());
        cookie.setMaxAge(30*60);
        cookie.setPath("/day07");//因为服务器写给浏览器的也有设置path,所以要想自己写的cookie覆盖服务器写的,必须跟服务器写的一致。
        response.addCookie(cookie);

5.Session是基于Cookie的,那如果浏览器禁用了cookie后怎么办? 会出现空指针异常。

因为禁用了cookie,服务器写cookie的时候,浏览器会拒绝,下次再来访问的时候是没有带cookie的,而在后台有使用这个cookie,所以会出现空指针异常。

解决方法:不以cookie的方式把JSESSIONID带给服务器,而是把JSESSIONID以url的方式带到服务器,

即URL重写:URL重写解决了禁用cookie后,Session的共享问题。

String url  = response.encodeRedirectURL("/day07/servlet/ListCartServlet");    //实现 url重写
response.sendRedirect(url);

encodeRedirectURL方法会自动的在URL后面加上JSEEIONID的参数。

6.一般涉及到会话的超链接都要使用url重写,来解决浏览器禁用cookie的问题。

String url = "/day07/servlet/BuyServlet?id="+book.getId();
url = response.encodeURL(url);  //得到重写后的url

encodeURL
encodeRedirectURL

上面两个都是url重写,重定向的url重写使用下面一个,其他的url重写使用上面一个。

response. encodeRedirectURL(java.lang.String url)
用于对sendRedirect方法后的url地址进行重写。
response. encodeURL(java.lang.String url)
用于对表单action和超链接的url地址进行重写

URL重写会自己判断浏览器是否禁用了cookie,如果没有禁用cookie,不会在超链接后面加上JSESSIONID的参数带给服务器,只有禁用了Cookie才会带。

Session的失效时间可以在web.xml文件中修改

  10 //10分钟

session在10分钟内没有在被使用,服务器就会摧毁session。

用户登录保存登录标记,存入session

public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        //request.getAttribute("username");  //从域中取username
        String username = request.getParameter("username");
        String password = request.getParameter("password");

        //检测用户名和密码在数据库中是否存在
        List<User> list = DB.getAll();
        for(User user : list){
            if(user.getUsername().equals(username)  && user.getPassword().equals(password)){
                //从数据库中找到匹配用户,让用户登陆成功
                request.getSession().setAttribute("user", user);
                response.sendRedirect("/day07/index.jsp");
                return;
            }
        }
        response.setCharacterEncoding("UTF-8");
        response.setContentType("text/html;charset=UTF-8");
        response.getWriter().write("用户名或密码错误!!");
    }

//注销登陆用户

public class LogoutServlet extends HttpServlet {

    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        HttpSession session = request.getSession(false);
        if(session!=null){
            session.removeAttribute("user");
        }

        response.sendRedirect("/day07/index.jsp");
    }

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

}

两者区别:

request.getSession();如果服务器内存中没有session,服务器会创建一个session,如果有session,会返回该session
request.getSession(false);只检索服务器内存中有没有session,如果没有也不创建session了,如果有,则返回该session

一个简单的例子

package session;


import java.io.IOException;
import java.io.PrintWriter;
import java.util.HashMap;
import java.util.Map;


import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;


public class CartServlet extends HttpServlet {


/**
* 
*/
private static final long serialVersionUID = 1L;


/**
* Constructor of the object.
*/
public CartServlet() {
super();
}


/**
* Destruction of the servlet. <br>
*/
public void destroy() {
super.destroy(); // Just puts "destroy" string in log
// Put your code here
}


/**
* The doGet method of the servlet. <br>
*
* This method is called when a form has its tag value method equals to get.
* 
* @param request the request send by the client to the server
* @param response the response send by the server to the client
* @throws ServletException if an error occurred
* @throws IOException if an error occurred
*/
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {


response.setContentType("text/html;UTF-8");

/**
* 1.购物车Map<String,Integer> cart 
* 2.获取购物车,判断是否第一次访问
* 3.继续购物或结算 
*/

//获取参数
String id = request.getParameter("id");

//购物车存入商品的名称和质量
String[] names = {"手电筒","冰箱","电视","洗衣机","电脑"};

//把id翻译名称
int idx = Integer.parseInt(id);
//商品名称
String name = names[idx-1];

//从session中获取购物车,先获取session
HttpSession session = request.getSession();

//从session中获取购物车
Map<String, Integer> cart = (Map<String, Integer>) session.getAttribute("cart");
//通过cart进行判断,是否是第一次访问
if(cart == null){
//创建购物车
cart = new HashMap<String, Integer>();
//第一次访问,放入名称和数量
cart.put(name, 1);
//存入到session中
session.setAttribute("cart", cart);
}else{
//是否包含改商品
if(cart.containsKey(name)){
//包含,取出数量加1加入购物车,存入session中
Integer count = cart.get(name);
count++;
//购物车存入商品
cart.put(name, count);
//存入的是购物车
session.setAttribute("cart", cart);
}else{
cart.put(name, 1);
}
}
//继续购物或结算
response.setContentType("text/html;UTF-8");
response.getWriter().write("<h3><a href='/day11/session/cartList.jsp'>继续购物</a><a href='/day11/session/pay.jsp'>去结算</a></h3>");
}


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


/**
* Initialization of the servlet. <br>
*
* @throws ServletException if an error occurs
*/
public void init() throws ServletException {
// Put your code here
}


}





<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>


<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <base href="<%=basePath%>">

    <title>My JSP 'cartList.jsp' starting page</title>

<meta http-equiv="pragma" content="no-cache">
<meta http-equiv="cache-control" content="no-cache">
<meta http-equiv="expires" content="0">    
<meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
<meta http-equiv="description" content="This is my page">
<!--
<link rel="stylesheet" type="text/css" href="styles.css">
-->


  </head>

  <body>
    <h3>手电筒<a href="/day11/cart?id=1">加入购物车</a></h3>
    <h3>冰箱<a href="/day11/cart?id=2">加入购物车</a></h3>
    <h3>电视<a href="/day11/cart?id=3">加入购物车</a></h3>
    <h3>洗衣机<a href="/day11/cart?id=4">加入购物车</a></h3>
    <h3>电脑<a href="/day11/cart?id=5">加入购物车</a></h3>
  </body>
</html>





<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>


<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <base href="<%=basePath%>">

    <title>My JSP 'pay.jsp' starting page</title>

<meta http-equiv="pragma" content="no-cache">
<meta http-equiv="cache-control" content="no-cache">
<meta http-equiv="expires" content="0">    
<meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
<meta http-equiv="description" content="This is my page">
<!--
<link rel="stylesheet" type="text/css" href="styles.css">
-->


  </head>

  <body>
    <h3>clear</h3>
    <%
    //获取购物车
    Map<String, Integer> cart = (Map<String, Integer>)request.getSession().getAttribute("cart");

      //购物车不为空
      if(cart != null){
      //
      Set<String> keys = cart.keySet();
      for(String key : keys){
   %>   

   <h3>您买的商品是<%= key %>,数量是<%= cart.get(key) %></h3>
    <%
      }
      }else{

     %> 
     <h3>您还没购物<a href="/day11/session/cartList.jsp">shopping</a></h3>
     <%
      }
     %>
  </body>
</html>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值