10-Cookie、Session
unit10-Cookie、Session
什么是会话
什么是会话:当浏览器发请求访问服务器开始,一直到访问服务器结束,浏览器关闭为止,这期间浏览器和服务器之间产生的所有请求和响应加在一起,就称之为浏览器和服务器之间的一次会话。
在一次会话中往往会产生一些数据,而这些数据往往是需要我们保存起来的,如何保存会话中产生的这些数据呢?
- 比如在购物过程中,将商品加入购物车,其实就是将商品信息保存到数据库中。(不讨论)
- 如果在没有登录时,将商品加入购物车,其实就是将商品信息保存到了cookie或session中。
可以使用cookie或者session保存会话中产生的数据。
如何将会话中产生的数据保存到cookie或者是session中?
cookie原理及应用
cookie的工作原理
35f0bd1bfc616e3bfcc9e96c189444ee
- Cookie是将会话中产生的数据保存在客户端,是客户端技术。
- Cookie是基于两个头进行工作的:分别是Set-Cookie响应头和Cookie请求头
- 通过Set-Cookie响应头将cookie从服务器端发送给浏览器,让浏览器保存到内部;而浏览器一旦保存了cookie,以后浏览器每次访问服务器时,都会通过cookie请求头,将cookie信息再带回服务器中。在需要时,在服务器端可以获取请求中的cookie中的数据,从而实现某些功能。
cookie的API及应用
1、创建Cookie对象
1 | Cookie c = new Cookie(String name, String value); |
2、将Cookie添加到response响应中
1 | response.addCookie( Cookie c ); |
3、获取请求中的所有cookie对象组成的数组
1 | Cookie[] cs = request.getCookies(); |
4、删除浏览器中的Cookie
1 | // cookie的API中没有提供直接删除cookie的方法,可以通过别的方式间接删除cookie |
代码示例:
1 | //创建一个名称为cart的cookie |
5、Cookie的常用方法
1 | cookie.getName(); // 获取cookie的名字 |
6、setMaxAge方法:设置cookie的最大生存时间
1 | 如果不设置该方法,cookie默认是会话级别的cookie,即生存时间是一次会话。当浏览器关闭,会话结束时,cookie也会被销毁(cookie默认存在浏览器的内存中,当浏览器关闭,内存释放,cookie也会随着内存的释放而销毁。) |
代码示例:
1 | //创建一个Cookie对象,将商品信息保存到cookie中 |
案例:使用cookie模拟购物车
1.index.html
1 | ... |
2.CartServlet
1 | protected void doGet(HttpServletRequest request, HttpServletResponse response) |
3.PayServlet
1 | protected void doGet(HttpServletRequest request, HttpServletResponse response) |
session原理及应用
session的工作原理
9f99b30f96234a6bba48e37d6852a080
- Session是将会话中产生的数据保存在服务器端,是服务器端技术
- Session是一个域对象,session中也保存了一个map集合,往session中存数据,其实就是将数据保存到session的map集合中。
- 通过session.setAttribute()方法可以将数据保存到session中,通过session.getAttribute()方法可以将数据从session中取出来。
session是一个域对象
获取session对象:
1 | request.getSession() |
Session是一个域对象,因此session中也提供了存取数据的方法。
1 | session.setAttribute(String attrName, Object attrValue); |
Session域对象的三大特征:
(1)生命周期:
**创建session:**第一次调用request.getSession()方法时,会创建一个session对象。(当浏览器在服务器端没有对应的session时,调用request.getSession()方法服务器会创建一个session对象。)
销毁session:
超时销毁:默认情况下,当超过30分钟没有访问session,session就会超时销毁。(30分钟是默认时间,可以修改,但不推荐修改)
自杀:调用session的invalidate方法时,会立即销毁session。
意外身亡:当服务器非正常关闭时(硬件损坏,断电,内存溢出等导致服务器非正常关闭),session会随着服务器的关闭而销毁;
当服务器正常关闭,在关闭之前,服务器会将内部的session对象序列化保存到服务器的work目录下,变为一个文件。这个过程叫做session的钝化(序列化);再次将服务器启动起来,钝化着的session会再次回到服务器,变为服务器中的对象,这个过程叫做session的活化(反序列化)。
**(2)作用范围:**在一次会话范围内(获取到的都是同一个session对象)
**(3)主要功能:**在整个会话范围内实现数据的共享
案例:使用session模拟购物车
1、index.html
1 | <body> |
2、CartServlet
1 | protected void doGet(HttpServletRequest request, HttpServletResponse response) |
3、PayServlet
1 | protected void doGet(HttpServletRequest request, HttpServletResponse response) |
总结:两者的区别
Cookie和session都属于会话技术,都可以保存会话中产生的数据,但由于cookie和session的工作原理和特点不同,因此两者的应用场景也不一样。
Cookie的特点:
cookie是将会话中产生的数据保存在浏览器客户端, 是客户端技术(JS可以访问cookie)
cookie是将数据保存在客户端浏览器,容易随着用户的操作导致cookie丢失或者被窃取,因此cookie中保存的数据不太稳定,也不太安全。
但cookie将数据保存在客户端,对服务器端没有太多影响,可以将数据保存很长时间。
总结:因此cookie中适合存储需要长时间保存、但对安全性要求不高的数据。
浏览器对cookie的大小和个数都有限制,一般推荐每一个站点给浏览器发送的cookie数量不超过20个,每一个cookie的大小不超过1kb。
Cookie的应用:实现购物车、记住用户名、30天内自动登录等。
Session的特点
- session是将会话中产生的数据保存在服务器端,是服务器端技术
- session将数据存在服务器端的session对象中,相对更加的安全,而且更加稳定。不容易随着用户的操作而导致session中的数据丢失或者是被窃取。
- 但session是服务器端的对象,在并发量较高时每一个浏览器客户端在服务器端都要对应一个session对象,占用服务器的内存空间,影响效率。
- 总结:因此session中适合存储对安全性要求较高,但不需要长时间保存的数据。
- Session的应用:保存登录状态、保存验证码
扩展内容
cookie中保存中文数据的问题
以下问题是针对Tomcat8.0及8.0以下的版本,在Tomcat8.5及8.5以后的版本中已经解决了该问题!
HTTP协议中规定了请求信息和响应信息中不能包含中文数据!
因此通过浏览器向服务器发送中文数据时,浏览器会将中文数据进行URL编码,编码为下面这种格式:
http://localhost/day13-cookie/index.html?user=%E5%BC%A0%E9%A3%9E%E9%A3%9E
将中文数据转成下面这种格式,叫做URL编码:
张飞飞 ---> URL编码 ---> %E5%BC%A0%E9%A3%9E%E9%A3%9E
将下面这种格式再次转回中文数据,叫做URL解码:
%E5%BC%A0%E9%A3%9E%E9%A3%9E ---> URL解码---> 张飞飞
问题:当cookie中保存中文数据,将cookie添加到响应中时,会报一个500异常,如下:
解决方法是:
将存入cookie中的先进行URL编码,再存入Cookie中,例如:
从cookie取出来的数据是进行URL编码后的数据,在使用之前需要进行URL解码,例如:
获取不到之前的session的问题
将商品保存到session中后,关闭浏览器再打开浏览器,访问服务器,此时获取不到之前的session。因为session是基于Cookie工作的。
在服务器创建一个session后,会为session分配一个独一无二的编号,称之为session的id,在此次响应时,服务器会将session的id以一个名称为JSESSIONID的cookie发送给浏览器保存到浏览器内部。
由于保存sessionid的cookie默认是会话级别的cookie,在浏览器关闭后,cookie会跟着销毁,sessionid也丢失了。因此下次访问服务器,没有session的id就获取不到之前的session。也获取不到session中的商品信息
解决方法:我们可以创建一个名称为JSESSIONID的cookie,其中保存session的ID,并设置cookie的最大存活时间,让cookie保存到硬盘上(即使浏览器关闭,cookie也不会销毁),这样下次访问服务器时,还可以将sessionid带给服务器,服务器可以通过sessionid获取到之前的session。 从session中获取到商品信息
9e195f495be5cc6e805c1185fcc753ad