## 会话技术:举例我和张三在路上碰面 我问他今天天气怎样他说挺好的 ,我又问你吃了吗他说你好烦哦又走了,这样就会话结束了。客户端与服务器端的
1. 会话:一次会话中包含多次请求和响应。
* 什么叫做一次会话:浏览器第一次给服务器资源发送请求,会话建立,直到服务器或者客户端一方断开为止
2. 功能:在一次会话的范围内的多次请求间,共享数据好比京东中买笔记本加入购物车 加入电源购物车 再点进入购物车结算就是一次会话范围内共享数据
3. 两种方式共享数据:
1. 客户端会话技术:Cookie
2. 服务器端会话技术:Session
## Cookie:
1. 概念:客户端会话技术,将数据保存到客户端 ------多次请求响应时第一次浏览器请求,服务器会携带一些数据相应给浏览器,浏览器会把这些数据保存在浏览器本地,下一次请求时浏览器会带着这些数据去发送请求,就意味着这两次请求响应存在数据共享
2. 快速入门:
* 使用步骤:第一个servlet
1. 创建Cookie对象,绑定数据
* new Cookie(String name, String value)
2. 发送Cookie对象
* response.addCookie(Cookie cookie)另一个servlet
3. 获取Cookie,拿到数据
* Cookie[] request.getCookies() 因为可能多个cookie,所以是个数组@WebServlet("/cookieDemo01") public class CookieDemo01 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //1.创建cookie对象 绑定数据 Cookie c = new Cookie("mag", "hello"); //2.发送cookie response.addCookie(c); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } } --------------------------------------------------------- @WebServlet("/cookieDemo02") public class CookieDemo02 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //3.获取cookie Cookie[] cs = request.getCookies(); //4.获取数据遍历cs if (cs!=null){ for (Cookie c : cs) { String name = c.getName(); //获取 name String value = c.getValue(); //获取value System.out.println(name+"--"+value); } } } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } }
先访问demo01 浏览器第一次请求
无反应 建立对话连接 (因为对话中没有跳转)
再访问demo02 相当于浏览器第二次请求
输出:
mag--hello
Idea-a933337d--f2ff2c05-c9c6-42e7-ad08-9ec9eb04c540如果直接再用新浏览器访问demo2 没有任何信息打印 -----------------注意
3. 实现原理
3. 实现原理
基于http协议来完成的 例如上图的两次请求响应首先客户端浏览器发送了第一次请求给服务器请求demo1的资源,demo1发送了cookie给客户端(cookie中保存的是msg-hello,其实是response会弄一个响应头
set-cookie:msg=hello)浏览器会收到这个响应头,浏览器会自动将头里面携带的数据保存在客户端浏览器中,保存之后下一次再次发送请求会数据会放在请求头里面
(cookie:msg=hello),可以通过api来获取。
* 基于响应头set-cookie和请求头cookie实现
4. cookie的细节
4. cookie的细节
1. 一次可不可以发送多个cookie?
* 可以
* 可以创建多个Cookie对象,使用response调用多次addCookie方法发送cookie即可。其实不只是访问demo2带cookie 访问demo1也带 只是没有获取
@WebServlet("/cookieDemo03") public class CookieDemo03 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //1.创建cookie对象 绑定数据 Cookie c1= new Cookie("mag", "hello"); Cookie c2 = new Cookie("name", "tt"); Cookie c3 = new Cookie("name", "yy"); //2.发送cookie response.addCookie(c1); response.addCookie(c2); response.addCookie(c3); } ----------------------------------------------------- @WebServlet("/cookieDemo02") public class CookieDemo02 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //3.获取cookie Cookie[] cs = request.getCookies(); //4.获取数据遍历cs if (cs!=null){ for (Cookie c : cs) { String name = c.getName(); String value = c.getValue(); System.out.println(name+"--"+value); } } }
2. cookie在浏览器中保存多长时间?
1. 默认情况下,当浏览器关闭后,Cookie数据被销毁,就是保存在浏览器内存中
2. 持久化存储:
* setMaxAge(int seconds)
1. 正数:将Cookie数据写到硬盘的文件中。持久化存储。并指定cookie存活时间,时间到后,cookie文件自动失效
2. 负数:默认值,当浏览器关闭后,Cookie数据被销毁
3. 零:删除cookie信息
设置30:先访问demo04,关闭浏览器,三十秒内再访问demo02,还能获得cookie设置-1: 先访问demo04,直接访问demo02有输出,关闭浏览器,再访问demo02没有输出
设置0: 好比先设置300s 访问demo04 此时300s内访问demo02都会打印信息,此时再设置0就是把浏览器中的cookie删除掉,300s内再访问demo02不会打印信息
3. cookie能不能存中文?
* 在tomcat 8 之前 cookie中不能直接存储中文数据。
* 需要将中文数据转码---一般采用URL编码(%E3)
* 在tomcat 8 之后,cookie支持中文数据。特殊字符还是不支持,建议使用URL编码存储,URL解码解析
4. cookie共享问题?
1. 假设在一个tomcat服务器中,部署了多个web项目,那么在这些web项目中cookie能不能共享?
* 默认情况下cookie不能共享
* setPath(String path):设置cookie的获取范围。默认情况下,设置当前的虚拟目录
* 如果要共享,则可以将path设置为"/"在一个tomcat服务器中,部署了多个web项目演示: 注意这是同一个tomcat部署了两个项目而且虚拟地址不一样的
演示
首先不设置"/"
day25下的demo05 设置cookie
存储cookie信息到浏览器:
day26下的demo 获取cookie到服务器
设置"/"
存储cookie
获取cookie
2. 不同的tomcat服务器间cookie共享问题?
例如百度新闻和百度贴吧是部署在不同的服务器上,要使用一级域名(.baidu.com)
* setDomain(String path):如果设置一级域名相同,那么多个服务器之间cookie可以共享
* setDomain(".baidu.com"),那么tieba.baidu.com和news.baidu.com中cookie可以共享
5. Cookie的特点和作用
5. Cookie的特点和作用
1. cookie存储数据在客户端浏览器
2. 浏览器对于单个cookie 的大小有限制(4kb) 以及 对同一个域名下的总cookie数量也有限制(20个)* 作用:
1. cookie一般用于存出少量的不太敏感的数据
2. 在不登录的情况下,完成服务器对客户端的身份识别(其实就是好比没有登录百度账号也能进行浏览器设置,关闭浏览器再打开还是保存这些设置,浏览器会以cookie的形式存储这些设置信息,而如果登陆了会把这些信息存储在服务器)
6.cookie案例
1. 需求:
1. 访问一个Servlet,如果是第一次访问,则提示:您好,欢迎您首次访问。
2. 如果不是第一次访问,则提示:欢迎回来,您上次访问时间为:显示时间字符串2. 分析:
1. 可以采用Cookie来完成
2. 在服务器中的Servlet判断是否有一个名为lastTime的cookie
1. 有:不是第一次访问
1. 响应数据:欢迎回来,您上次访问时间为:2018年6月10日11:50:20
2. 写回Cookie:lastTime=2018年6月10日11:50:01
2. 没有:是第一次访问
1. 响应数据:您好,欢迎您首次访问
2. 写回Cookie:lastTime=2018年6月10日11:50:01代码:
@WebServlet("/cookieTest02") public class cookieTest02 extends HttpServlet { protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { //响应消息体为汉字需要设置数据格式以及编码: response.setContentType("text/html;charset=utf-8"); //1获取所有cookies Cookie[] cs = request.getCookies(); //定义一个标记 boolean flag=false; //没有cookie为lastTime //2遍历cookies数组 if (cs!=null&&cs.length>0){ for (Cookie c : cs) { ///3.获取cookie的名称 String name = c.getName(); //4.判断名称是否是lastTime if ("lastTime".equals(name)){ //5 true 该cookie表示不是第一次访问 flag=true; //有lastTime的cookie //设置cookie的value //获取当前时间的字符串 重新设置cookie的值 重新发送cookie Date date = new Date(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss"); //这里空格会报500 需要url编码解码 tomcat8对特殊字符还是不支持建议使用url编码存储 url解码解析 String str_date = sdf.format(date); System.out.println("编码前"+str_date); //URL编码 str_date = URLEncoder.encode(str_date, "utf-8"); //注意等号前面的String没有 这样就可以重名了参数里的参数名称 System.out.println("编码后"+str_date); c.setValue(str_date);//设置lastTime的value //设置cookie的存活时间为30天 c.setMaxAge(60*60*24*30); //重新发送value response.addCookie(c); //5.1 先响应数据 //5.1.1获取cookie的value 就是一个时间 String value = c.getValue(); System.out.println("解码前"+value); //URL解码 value= URLDecoder.decode(value,"utf-8"); System.out.println("解码后"+value); response.getWriter().write("<h1>欢迎回来,您上次的访问时间是"+value+"</h1>"); break;//找到一个就不要循环了,直接关闭循环 } } } if (cs==null||cs.length==0||flag==false){ //表示没有lastTime的cookie 是第一次访问 //先设置cookie的value Date date = new Date(); SimpleDateFormat sdf = new SimpleDateFormat("yyyy年MM月dd日 HH:mm:ss"); String str_date = sdf.format(date); System.out.println("编码前"+str_date); //URL编码 str_date = URLEncoder.encode(str_date, "utf-8"); //注意等号前面的String没有 这样就可以重名了参数里的参数名称 System.out.println("编码后"+str_date); Cookie c = new Cookie("lastTime",str_date); c.setMaxAge(60*60*30*30); response.addCookie(c); response.getWriter().write("<h1>您好!欢迎首次访问</h1>"); } } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { this.doPost(request, response); } }
进行了持久化的存储就算是再次访问还是有记录