## 会话技术:举例我和张三在路上碰面 我问他今天天气怎样他说挺好的 ,我又问你吃了吗他说你好烦哦又走了,这样就会话结束了。客户端与服务器端的 ?? ?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);
}
}
?
进行了持久化的存储就算是再次访问还是有记录
|