一、Cookie的出现

浏览器和服务器之间的通信少不了HTTP协议,但是因为HTTP协议是无状态的,所以服务器并不知道上一次浏览器做了什么样的操作,这样严重阻碍了交互式Web应用程序的实现。

针对上述的问题,网景公司的程序员创造了Cookie。

二、Cookie的传输

服务器端在实现Cookie标准的过程中,需要对任意HTTP请求发送Set-Cookie HTTP头作为响应的一部分:

Set-Cookie: name=value; expires=Tue, 03-Sep-2019 14:10:21 GMT; path=/; domain=.xxx.com;

浏览器端会存储这样的Cookie,并且为之后的每个请求添加Cookie HTTP请求头发送回服务器:

Cookie: name=value

服务器通过验证Cookie值,来判断浏览器发送请求属于哪一个用户。


三、浏览器中的Cookie

浏览器中的Cookie主要由以下几部分组成:

  • 名称:Cookie唯一的名称,必须经过URL编码处理。(同名会出现覆盖的情况)
  • 值:必须经过URL编码处理。
  • 域(domain):默认情况下cookie在当前域下有效,你也可以设置该值来确保对其子域是否有效。
  • 路径(path):指定Cookie在哪些路径下有效,默认是当前路径下。
  • 失效时间(expires):默认情况下,浏览器会话结束时会自动删除Cookie;也可以设置一个GMT格式的日期,指定具体的删除日期;如果设置的日期为以前的日期,那么Cookie会立即删除。
  • 安全标志(secure):指定之后只允许Cookie发送给https协议。

浏览器在发送请求时,只会将名称与值添加到请求头的Cookie字段中,发送给服务端。

浏览器提供了一个非常蹩脚的API来操作Cookie:

document.cookie

通过上述方法可以对该Cookie进行写操作,每一次只能写入一条Cookie字符串:

document.cookie = 'a=1; secure; path=/'

通过该方法还可以进行Cookie的读操作:

document.cookie   // "a=1"

由于上述方法操作Cookie非常的不直观,一般都会写一些函数来简化Cookie读取、设置和删除操作。

对于Cookie的设置操作中,需要以下几点:

  • 对于名称和值进行URL编码处理,也就是采用JavaScript中的encodeURIComponent()方法;
  • expires要求传入GMT格式的日期,需要处理为更易书写的方式,比如:设置秒数的方式;
  • 注意只有的属性名的secure;
  • 每一段信息需要采用分号加空格
function setCookie (key, value, attributes) {if (typeof document === 'undefined') {return}attributes = Object.assign({}, {path: '/'}, attributes)let { domain, path, expires, secure } = attributesif (typeof expires === 'number') {expires = new Date(Date.now() + expires * 1000)}if (expires instanceof Date) {expires = expires.toUTCString()} else {expires = ''}key = encodeURIComponent(key)value = encodeURIComponent(value)let cookieStr = `${key}=${value}`if (domain) {cookieStr += `; domain=${domain}`}if (path) {cookieStr += `; path=${path}`}if (expires) {cookieStr += `; expires=${expires}`}if (secure) {cookieStr += `; secure`}return (document.cookie = cookieStr)
}

Cookie的读操作需要注意的是将名称与值进行URL解码处理,也就是调用JavaScript中的decodeURIComponent()方法:

function getCookie (name) {if (typeof document === 'undefined') {return}let cookies = []let jar = {}document.cookie && (cookies = document.cookie.split('; '))for (let i = 0, max = cookies.length; i < max; i++) {let [key, value] = cookies[i].split('=')key = decodeURIComponent(key)value = decodeURIComponent(value)jar[key] = valueif (key === name) {break}}return name ? jar[name] : jar
}

最后一个清除的方法就更加简单了,只要将失效日期(expires)设置为过去的日期即可:

function removeCookie (key) {
setCookie(key, '', { expires: -1 }) }

介绍Cookie基本操作的封装之后,还需要了解浏览器为了限制Cookie不会被恶意使用,规定了Cookie所占磁盘空间的大小以及每个域名下Cookie的个数。

为了绕开单域名下Cookie个数的限制,开发人员还创造了一种称为subcookie的概念,这里就不在赘述了,可以参考【JavaScript高级程序设计第23章 p633】。

四、服务端的Cookie

相比较浏览器端,服务端执行Cookie的写操作时,是将拼接好的Cookie字符串放入响应头的Set-Cookie字段中;执行Cookie的读操作时,则是解析HTTP请求头字段Cookie中的键值对。

与浏览器最大的不同,在于服务端对于Cookie的安全性操碎了心

signed

当设置signed=true时,服务端会对该条Cookie字符串生成两个Set-Cookie响应头字段:

  Set-Cookie: lastTime=2019-03-05T14:31:05.543Z; path=/; httponlySet-Cookie: lastTime.sig=URXREOYTtMnGm0b7qCYFJ2Db400; path=/; httponly

这里通过再发送一条以.sig为后缀的名称以及对值进行加密的Cookie,来验证该条Cookie是否在传输的过程中被篡改。

httpOnly

服务端Set-Cookie字段中新增httpOnly属性,当服务端在返回的Cookie信息中含有httpOnly字段时,开发者是不能通过JavaScript来操纵该条Cookie字符串的。

这样做的好处主要在于面对XSS(Cross-site scripting)攻击时,黑客无法拿到设置httpOnly字段的Cookie信息。

此时,你会发现localStorage相比较Cookie,在XSS攻击的防御上就略逊一筹了。
sameSite

在介绍这个新属性之前,首先你需要明白:当用户从http://a.com发起http://b.com的请求也会携带上Cookie,而从http://a.com携带过来的Cookie称为第三方Cookie。

虽然第三方Cookie有一些好处,但是给CSRF(Cross-site request forgrey)攻击的机会。

为了从根源上解决CSRF攻击,sameSite属性便闪亮登场了,它的取值有以下几种:

  • strict:浏览器在任何跨域请求中都不会携带Cookie,这样可以有效的防御CSRF攻击,但是对于有多个子域名的网站采用主域名存储用户登录信息的场景,每个子域名都需要用户重新登录,造成用户体验非常的差。
  • lax:相比较strict,它允许从三方网站跳转过来的时候使用Cookie。

为了方便大家理解sameSite的实际效果,可以看这个例子:

 // a.com 服务端会在访问页面时返回如下Cookiecookies.set('foo', 'aaaaa')cookies.set('bar', 'bbbbb')cookies.set('name', 'cccccc')// b.com 服务端会在访问页面时返回如下Cookiecookies.set('foo', 'a', { sameSite: 'strict' })cookies.set('bar', 'b', { sameSite: 'lax' })cookies.set('baz', 'c')

如何现在用户在a.com中点击链接跳转到b.com,它的请求头是这样的:

Request Headers Cookie: bar=b; baz=c

五、网站性能优化

Cookie在服务端和浏览器的通信中,主要依靠HTTP的响应头和请求头传输的,所以Cookie会占据一定的带宽。

前面提到浏览器会为每一次HTPP请求自动携带上Cookie信息,但是对于同站内的静态资源,服务器并不需要处理其携带的Cookie,这无形中便浪费了带宽。

在最佳实践中,一般都会将静态资源部署到独立的域名上,从而可以避免无效Cookie的影响。


作者:descire

链接:http://www.imooc.com/article/286535

Cookie中不能有空格_前端小贴士 -- 全面了解Cookie相关推荐

  1. Cookie中不能有空格_PHP 使用cookie

    1.把数据写入cookie中 setcookie()函数 setcookie(name,value,expire,path,domain,secure) 必须在任何其他输出发送到客户端前对 cooki ...

  2. html 去文本框中的双引号_前端·HTML基础

    知识点1:HTML.XHTML与HTML5 HTML(超文本标记语言)是一种用来描述网页的标记语言(不是编程语言).其结构包括head部分和body部分,前者存放网页信息,后者提供具体内容. XHTM ...

  3. git 如何忽略掉文件夹_#PY小贴士# 我的git仓库为什么每次提交都有很多改动?

    git 是如今最流行的代码版本管理工具,没有之一. 今天说一个 git 使用时的细节:.gitignore 我们在使用 git 管理代码时,如果默认把项目里的所有文件都 add 进去,加入到仓库中,会 ...

  4. 微信小程序wxml如何判断字符串中汉语某字符_微信小程序开发经典案例解析“嗨兔儿”...

    嗨兔儿是微信公众号嗨日语歌(hitaici)助手,主要为用户提供,关键词检索,帮助手册等,为外语学习者提供一个便捷的操作方式,能够开心工作,开心生活. 开发过程及注意事项分享视频. 1. 微信小程序开 ...

  5. es6 去掉空格_微信小程序自动去除input空格的方法

    当用户输入账号或者密码的时候,可能会有输入空格的情况,但是实际需要是不能够有空格的,所以我们要做一个去除空格,并且适应所有input的name参数的方法,下面请看源码: wxml: js: verif ...

  6. python中拼音怎么用_实用小技巧,Python一秒将汉字转为拼音

    大家好,欢迎来到Crossin的编程教室! 有时在处理文件时候,我们需要将中文汉字转换为拼音或者拼音首字母的需求:还有的时候,我们需要对汉字进行按拼音字母排序(汉字的默认排序是编码序而非拼音序).针对 ...

  7. java socket中属性详解_前端开发:关于Vue组件中的data属性值是函数而不是对象的详解...

    最近在搞关于前端开发的基础知识归纳,发现了不少经典知识点,那么本篇博文就来分享一个经典的知识点:Vue组件中的data属性值为什么是函数而不是对象.首先来了解一下Vue组件的使用理念:在Vue组件使用 ...

  8. 前端怎么画三角形_前端小技巧:边框写三角形

    边框写三角形,这个技能从事前端的小伙伴应该都不陌生,不过大多数都是知其然而不知其所以然,今天我们就来探究一下这里面的原理. 随意的在页面上给一个盒子,然后给一个边框来看看. 这样的一段代码,给到盒子的 ...

  9. python控制代码块逻辑关系_Python 编程中用代码缩进表示逻辑递进关系,通常用几个空格_学小易找答案...

    [判断题]在自动化现场的跟踪过程中,钢板的数据定义可以定义为一个类(class),现场的每一块钢板称为一个实例. [多选题]下列Python语句正确的是 [判断题]四联球菌.八叠球菌.葡萄球菌均是多细 ...

最新文章

  1. DefaultTableCellRenderer
  2. MB_SELECT_GR_BLOCKED_STOCK 读取物料收货冻结库存
  3. 《ASP.NET MVC企业实战》(一) MVC开发前奏
  4. Vs code 02 配置非标准的C、C++的编译环境
  5. 临时上传的文件-20170707
  6. 25% 的开发者认为 Rust 是最佳替代,最新 Go 开发者调查报告出炉
  7. java随机数函数生成指定区间的,意外的惊喜
  8. xlsread服务器出现意外情况
  9. 图解机器学习算法(10) | XGBoost模型最全解析(机器学习通关指南·完结)
  10. matlab:快速傅里叶(反)变换 FFTIFFT
  11. 课题:交通标志识别——设计思路与实现步骤记录
  12. 【软著】自己申请软件著作权流程
  13. 【Unity】 HTFramework框架(十七)Hotfix热更新模块
  14. 利用费马小定理进行素性测试
  15. Wechat + unbuntu
  16. 电子邮件群发最好用的邮箱是哪个?
  17. (1)输入直角三角形的两个直角边的长度a,b,求斜边c的长度 (2) 编写一个程序,用于两个数的交换
  18. oracle-ORA的原理
  19. 弥散磁共振影像处理的黎曼 芬斯勒几何方法研究
  20. [Unity3D]Unity3D游戏开发之继续探索NGUI

热门文章

  1. 打造大型直播平台之《PC客户端架构》
  2. yxcms安装环境php,Windows7下PHP开发环境安装配置图文方法
  3. python函数和模块的使用方法_Python学习06_函数和模块的使用
  4. 生成私钥 p q rsa_【安全】理解 RSA 算法
  5. kotlin自定义View出现 java.lang.ClassNotFoundException
  6. poe交换机的作用和优点
  7. 485转换器产品类型介绍
  8. [渝粤教育] 西南科技大学 电子产品制造工艺 在线考试复习资料
  9. 【渝粤题库】国家开放大学2021春2109刑事诉讼法学题目
  10. 【案例】无线串口模块 星型网络组网 案例