详解HTTP Cookie状态管理机制


Posted in PHP onJanuary 14, 2016

HTTP cookies,通常又称作"cookies",已经存在了很长时间,但是仍旧没有被予以充分的理解。首要的问题是存在了诸多误区,认为cookies是后门程序或病毒,或压根不知道它是如何工作的。第二个问题是对于cookies缺少一个一致性的接口。尽管存在着这些问题,cookies仍旧在web开发中起着如此重要的作用,以至于如果cookie在没有可替代品出现的情况下消失,我们许多喜欢的Web应用将变得毫无用处。

一、cookie 起源

cookie 最早是网景公司的雇员 Lou Montulli 在1993年3月发明,后被 W3C 采纳,目前 cookie 已经成为标准,所有的主流浏览器如 IE、Chrome、Firefox、Opera 等都支持。

cookie 的诞生是由于 HTTP 协议的天生缺陷,HTTP 是一种无状态的协议,简单的 Request 和 Response 一旦请求/响应结束,客户端与服务器端的连接就会关闭,再次交换数据需要建立新的连接。这就意味着服务器无法从连接上跟踪会话,即服务器并不清楚是哪个客户端。

一些典型应用如 登陆/购物车 就无法实现了。比如,用户 A 在购物商城购买的商品都应该放在 A 的购物车内,不论是用户 A 什么时间购买的,这都是属于同一个会话的,不能放入用户 B 或用户 C 的购物车内,这不属于同一个会话。

基本的原理如图

详解HTTP Cookie状态管理机制

二、cookie 操作

对 cookie 的操作包括如下

1.名称(Name)
2.值(Value)
3.域(Domain)
4.路径(Path)
5.失效日期(Expires)
6.安全标志(Secure)
7.HttpOnly (仅服务器端)

注意,cookie 多数时候由服务器端创建,JS 也可以创建 cookie,但 HttpOnly 类型的 JS 无法创建。

浏览器提供的 cookie API (document.cookie)实在过于简陋,可以稍封装下,如以下采用setter/getter方式 cookie 函数就方便了许多

/*
* JS 写cookie和读cookie操作
*
* **取cookie**
* cookie(name)
*
* **写cookie**
* cookie(name, value)
* cookie(name, value, option)
*/
var cookie = function(name, value, option) {
var doc = document
if (value != undefined) { // set 
option = option || {}
if (value === null) {
value = ''
option.expires = -1
}
var expires = ''
if (option.expires && (typeof option.expires == 'number' || option.expires.toUTCString)) {
var date = new Date
if (typeof option.expires == 'number') {
date.setTime(date.getTime() + (option.expires * 24 * 60 * 60 * 1000))
} else {
date = option.expires
}
// for IE
expires = '; expires=' + date.toUTCString()
}
var path = option.path ? '; path=' + option.path : ''
var domain = option.domain ? '; domain=' + option.domain : ''
var secure = option.secure ? '; secure' : ''
doc.cookie = [name, '=', encodeURIComponent(value), expires, path, domain, secure].join('')
} else { // get 
var cookieValue = null
if (doc.cookie && doc.cookie != '') {
var cookies = doc.cookie.split(';')
for (var i = 0; i < cookies.length; i++) {
var cookie = $.trim(cookies[i]).split('=')
if ( cookie[0] == name && cookie.length > 1 ) {
try {
cookieValue = decodeURIComponent(cookie[1])
} catch(e) {
cookieValue = cookie[1]
}
break
}
}
}
return cookieValue
}
};

当然,还有更方便的 https://github.com/florian/cookie.js,提供了更多便捷函数。

三、cookie 类型

1.普通 cookie,服务器端和 JS 都可以创建,JS 可以访问
2.HttpOnly cookie,只能由服务端创建,JS 是无法读取的,主要基于安全考虑
3.安全的 cookie (仅https),服务器端和 JS 都可以创建,JS 仅HTTPS下访问

比如,在新浪云上测试页面:http://snandy.sinaapp.com/php/cookie.php,我种了 3 个 cookie,分别是 c1, c2, c3

$d1 = mktime(1,1,1,1,1,2018);
// 普通cookie
setcookie("c1", "Jack", $d1); 
// 安全的cookie,仅https,第6个参数
setcookie("c2", "John", $d1, NULL, NULL, TRUE); 
// HttpOnly cookie 第7个参数
setcookie("c3", "Resig", $d1, NULL, NULL, NULL, TRUE);

用 Firefox 访问

详解HTTP Cookie状态管理机制

我种的三个都有,saeut是新浪云种的。

在 firebug 控制台输入 document.cookie

详解HTTP Cookie状态管理机制

可以看到,c2,c3 都是访问不到的。c2 是 安全的cookie,需要在https协议下访问,c3 则是 httpOnly 的,JS无法访问,这个需要注意。

把访问协议改成 https: https://snandy.sinaapp.com/php/cookie.php,firebug 切换到控制台再输入 document.cookie,可以看到 c2 就可以访问了

详解HTTP Cookie状态管理机制

四、cookie 的坑

详解HTTP Cookie状态管理机制

1. Cookie 太大或数量过多时页面访问报错,比如会出现如下提示

因此站点的 cookie 需要管理,不能随意种 cookie。另外尽量指定path,将cookie限定在指定范围内。

网站 browsercookielimits.squawky.net ,记录了各浏览器 cookie 大小

详解HTTP Cookie状态管理机制

2. 保存中文时需要Unicode编码(encodeURIComponent),否则存的是乱码

PHP 相关文章推荐
phpMyadmin 用户权限中英对照
Apr 02 PHP
PHP下escape解码函数的实现方法
Aug 08 PHP
PHP数组无限分级数据的层级化处理代码
Dec 29 PHP
Apache实现Web Server负载均衡详解(不考虑Session版)
Jul 05 PHP
体育彩票排列三组选三算法分享
Mar 07 PHP
PHP实现利用MySQL保存session的方法
Aug 23 PHP
PHP+MySQL删除操作实例
Jan 21 PHP
php实现scws中文分词搜索的方法
Dec 25 PHP
PHP简单数据库操作类实例【支持增删改查及链式操作】
Oct 10 PHP
Thinkphp 5.0实现微信企业付款到零钱
Sep 30 PHP
YII分模块加载路由的实现方法
Oct 01 PHP
php 文件上传至OSS及删除远程阿里云OSS文件
Jul 04 PHP
在php中设置session用memcache来存储的方法总结
Jan 14 #PHP
thinkphp实现图片上传功能
Jan 13 #PHP
PHP实现伪静态方法汇总
Jan 13 #PHP
微信公众号支付之坑:调用支付jsapi缺少参数 timeStamp等错误解决方法
Jan 12 #PHP
优化WordPress中文章与评论的时间显示
Jan 12 #PHP
win平台安装配置Nginx+php+mysql 环境
Jan 12 #PHP
WordPress中重置文章循环的rewind_posts()函数讲解
Jan 11 #PHP
You might like
apache+codeigniter 通过.htcaccess做动态二级域名解析
2012/07/01 PHP
浅谈php的优缺点
2015/07/14 PHP
TP5(thinkPHP框架)实现后台清除缓存功能示例
2019/05/29 PHP
javascript延时重复执行函数 lLoopRun.js
2007/06/29 Javascript
jQuery插件原来如此简单 jQuery插件的机制及实战
2012/02/07 Javascript
多次注册事件会导致一个事件被触发多次的解决方法
2013/08/12 Javascript
浏览器的JavaScript引擎的识别方法
2013/10/20 Javascript
JQuery 控制内容长度超出规定长度显示省略号
2014/05/23 Javascript
js简单判断flash是否加载完成的方法
2016/06/21 Javascript
JavaScript中最容易混淆的作用域、提升、闭包知识详解(推荐)
2016/09/05 Javascript
老生常谈原生JS执行环境与作用域
2016/11/22 Javascript
关于不同页面之间实现参数传递的几种方式讨论
2017/02/13 Javascript
jQuery插件FusionCharts实现的MSBar3D图效果示例【附demo源码】
2017/03/23 jQuery
原生JS实现的放大镜特效示例【测试可用】
2018/12/08 Javascript
Vue中的循环及修改差值表达式的方法
2019/08/29 Javascript
小程序实现录音上传功能
2019/11/22 Javascript
原生js实现分页效果
2020/09/23 Javascript
[01:09:13]DOTA2-DPC中国联赛 正赛 CDEC vs XG BO3 第三场 1月19日
2021/03/11 DOTA
pygame实现弹力球及其变速效果
2017/07/03 Python
解读! Python在人工智能中的作用
2017/11/14 Python
python获取代码运行时间的实例代码
2018/06/11 Python
python样条插值的实现代码
2018/12/17 Python
Python统计分析模块statistics用法示例
2019/09/06 Python
下载官网python并安装的步骤详解
2019/10/12 Python
Python:type、object、class与内置类型实例
2019/12/25 Python
在python tkinter界面中添加按钮的实例
2020/03/04 Python
详解pandas获取Dataframe元素值的几种方法
2020/06/14 Python
欧洲最大的品牌水上运动服装和设备在线零售商:Wuituit Outlet
2018/05/05 全球购物
全球性的奢侈品梦工厂:Forzieri(福喜利)
2019/02/20 全球购物
励志演讲稿大全
2014/08/21 职场文书
出国签证在职证明
2014/09/20 职场文书
部队2014年终工作总结
2014/11/27 职场文书
2015年初一班主任工作总结
2015/05/13 职场文书
消夏晚会主持词
2015/06/30 职场文书
学校2016年全国助残日活动总结
2016/04/01 职场文书
女人创业励志语录,句句蕴含能量,激发你的潜能
2019/08/20 职场文书