蚂蚁学堂(1):10-Session与Cokile实现原理

一、 Cooke技术

1.1 特点

Cookie技术:会话数据保存在浏览器客户端。

1.2 Cookie技术核心

Cookie类:用于存储会话数据

1)构造Cookie对象

      Cookie(java.lang.String name, java.lang.String value)

2)设置cookie

      void setPath(java.lang.String uri)   :设置cookie的有效访问路径

      void setMaxAge(int expiry) : 设置cookie的有效时间

      void setValue(java.lang.String newValue) :设置cookie的值

3)发送cookie到浏览器端保存

      void response.addCookie(Cookie cookie)  : 发送cookie

4)服务器接收cookie

     Cookie[] request.getCookies()  : 接收cookie

1.3 Cookie原理

 1)服务器创建cookie对象,把会话数据存储到cookie对象中。

                     new Cookie("name","value");

 2)  服务器发送cookie信息到浏览器

                     response.addCookie(cookie);

举例: set-cookie: name=eric  (隐藏发送了一个set-cookie名称的响应头)

3)浏览器得到服务器发送的cookie,然后保存在浏览器端。

4)浏览器在下次访问服务器时,会带着cookie信息 

举例: cookie: name=eric  (隐藏带着一个叫cookie名称的请求头)

 5)服务器接收到浏览器带来的cookie信息: 

                       request.getCookies();

1.4 Cookie的细节

1)void setPath(java.lang.String uri)   :设置cookie的有效访问路径。有效路径指的是cookie的有效路径保存在哪里,那么浏览器在有效路径下访问服务器时就会带着cookie信息,否则不带cookie信息。

2)void setMaxAge(int expiry) : 设置cookie的有效时间。

正整数:表示cookie数据保存浏览器的缓存目录(硬盘中),数值表示保存的时间。

负整数:表示cookie数据保存浏览器的内存中。浏览器关闭cookie就丢失了!!

零:表示删除同名的cookie数据

3)Cookie数据类型只能保存非中文字符串类型的。可以保存多个cookie,但是浏览器一般只允许存放300个Cookie,每个站点最多存放20个Cookie,每个Cookie的大小限制为4KB。

三、Session技术

3.1 引入

Cookie的局限:

             1)Cookie只能存字符串类型。不能保存对象

             2)只能存非中文。

             3)1个Cookie的容量不超过4KB。

如果要保存非字符串,超过4kb内容,只能使用session技术!!!

Session特点:

会话数据保存在服务器端。(内存中)

3.2 Session技术核心

HttpSession类:用于保存会话数据

1)创建或得到session对象

             HttpSession getSession()

            HttpSession getSession(boolean create)

2)设置session对象

          void setMaxInactiveInterval(int interval)  : 设置session的有效时间

         void invalidate()     : 销毁session对象

         java.lang.String getId()  : 得到session编号

3)保存会话数据到session对象

        void setAttribute(java.lang.String name, java.lang.Object value)  : 保存数据

        java.lang.Object getAttribute(java.lang.String name)  : 获取数据

        void removeAttribute(java.lang.String name) : 清除数据

3.3 Session原理

问题: 服务器能够识别不同的浏览者!!!

前提: 在哪个session域对象保存数据,就必须从哪个域对象取出!!!!

浏览器1:(给s1分配一个唯一的标记:s001,把s001发送给浏览器)

1)创建session对象,保存会话数据

       HttpSession session = request.getSession();   --保存会话数据 s1

       浏览器1 的新窗口(带着s001的标记到服务器查询,s001->s1,返回s1)

1)得到session对象的会话数据

        HttpSession session = request.getSession();   --可以取出  s1

新的浏览器1:(没有带s001,不能返回s1)

1)得到session对象的会话数据

      HttpSession session = request.getSession();   --不可以取出  s1

代码解读:HttpSession session = request.getSession();

1)第一次访问创建session对象,给session对象分配一个唯一的ID,叫JSESSIONID

          new HttpSession();

2)把JSESSIONID作为Cookie的值发送给浏览器保存

        Cookie cookie = new Cookie("JSESSIONID", sessionID);

       response.addCookie(cookie);

3)第二次访问的时候,浏览器带着JSESSIONID的cookie访问服务器

4)服务器得到JSESSIONID,在服务器的内存中搜索是否存放对应编号的session对象。

         if(找到){

               return map.get(sessionID);

        }

        Map<String,HttpSession>

5)如果找到对应编号的session对象,直接返回该对象

6)如果找不到对应编号的session对象,创建新的session对象,继续走1的流程

结论:通过JSESSIONcookie值在服务器找session对象!!!!!

总结:

 1)会话管理: 浏览器和服务器会话过程中的产生的会话数据的管理。

2)Cookie技术:

            new Cookie("name","value")

            response.addCookie(coookie)

            request.getCookies()

3)Session技术

           request.getSession();

getSession(true) / getSession()  : 创建或得到session对象。没有匹配的session编号,自动创建新的session对象。

getSession(false):                        得到session对象。没有匹配的session编号,返回null

           setAttrbute("name","会话数据");

           getAttribute("会话数据")

          void setMaxInactiveInterval(int interval)  : 设置session的有效时间

          session对象销毁时间:3.1 默认情况30分服务器自动回收

                                                3.2 修改session回收时间

                                                3.3 全局修改session有效时间

问题:如何避免浏览器的JSESSIONID的cookie随着浏览器关闭而丢失的问题

        // 手动发送一个硬盘保存的cookie给浏览器

       Cookie c = new Cookie("JSESSIONID",session.getId());

       c.setMaxAge(60*60);

       response.addCookie(c);

四、自定义Token

4.1 什么是token

token其实就是一个令牌,具有随机性,类似于sessionId。

在对接一些第三方平台的时候,为了能够保证数据安全性,通常会使用一些令牌进行交互

例如: https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1421140183

4.2 如何自定义token

token生成规则,只要保证token生成一个不重复的唯一字符串即可。

使用jdk自带的uuid生成规则

五、表单重复提交解决方案(防止Http重复提交。)

1.1网络延时

 在平时开发中,如果网速比较慢的情况下,用户提交表单后,发现服务器半天都没有响应,那么用户可能会以为是自己没有提交表单,就会再点击提交按钮重复提交表单,我们在开发中必须防止表单重复提交。

1.2重新刷新

表单提交后用户点击【刷新】按钮导致表单重复提交

1.3点击浏览器的【后退】按钮回退到表单页面后进行再次提交

用户提交表单后,点击浏览器的【后退】按钮回退到表单页面后进行再次提交

2.1 使用javascript 解决

 既然存在上述所说的表单重复提交问题,那么我们就要想办法解决,比较常用的方法是采用JavaScript来防止表单重复提交,具体做法如下:

修改form.jsp页面,添加如下的JavaScript代码来防止表单重复提交

function dosubmit(){
    //获取表单提交按钮
    var btnSubmit = document.getElementById("submit");
    //将表单提交按钮设置为不可用,这样就可以避免用户再次点击提交按钮
    btnSubmit.disabled= "disabled";
    //返回true让表单可以正常提交
    return true;
}

使用后端提交解决

具体的做法:在服务器端生成一个唯一的随机标识号,专业术语称为Token(令牌),同时在当前用户的Session域中保存这个Token。然后将Token发送到客户端的Form表单中,在Form表单中使用隐藏域来存储这个Token,表单提交的时候连同这个Token一起提交到服务器端,然后在服务器端判断客户端提交上来的Token与服务器端生成的Token是否一致,如果不一致,那就是重复提交了,此时服务器端就可以不处理重复提交的表单。如果相同则处理表单提交,处理完后清除当前用户的Session域中存储的标识号。
  在下列情况下,服务器程序将拒绝处理用户提交的表单请求:

  1. 存储Session域中的Token(令牌)与表单提交的Token(令牌)不同。
  2. 当前用户的Session中不存在Token(令牌)
  3. 用户提交的表单数据中没有Token(令牌)

 

Web安全

XSS攻击使用Javascript脚本注入进行攻击

例如在表单中注入: <script>location.href='http://www.itmayiedu.com'</script>

解决: 使用Fileter防止XSS攻击

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值