详解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 相关文章推荐
Zend studio for eclipse中使php可以调用mysql相关函数的设置方法
Oct 13 PHP
php access 数据连接与读取保存编辑数据的实现代码
May 12 PHP
PHP网页游戏学习之Xnova(ogame)源码解读(八)
Jun 23 PHP
ThinkPHP字符串函数及常用函数汇总
Jul 18 PHP
codeigniter实现get分页的方法
Jul 10 PHP
php及codeigniter使用session-cookie的方法(详解)
Apr 06 PHP
解决laravel 5.1报错:No supported encrypter found的办法
Jun 07 PHP
php安装扩展mysqli的实现步骤及报错解决办法
Sep 23 PHP
PHP设计模式之状态模式定义与用法详解
Apr 02 PHP
Yii2框架数据验证操作实例详解
May 02 PHP
PHP切割汉字的常用方法实例总结
Apr 27 PHP
Yii2处理密码加密及验证的方法
May 12 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
PHP实现全角字符转为半角方法汇总
2015/07/09 PHP
详解PHP中的mb_detect_encoding函数使用方法
2015/08/18 PHP
thinkPHP5.1框架路由::get、post请求简单用法示例
2019/05/06 PHP
仿163填写邮件地址自动显示下拉(无优化)
2008/11/05 Javascript
javascript 检测浏览器类型和版本的代码
2009/09/15 Javascript
jquery ajax对特殊字符进行转义防止js注入使用示例
2013/11/21 Javascript
如何实现修改密码时密码框显示保存到cookie的密码
2013/12/10 Javascript
8个超实用的jQuery功能代码分享
2015/01/08 Javascript
JavaScript的React框架中的JSX语法学习入门教程
2016/03/05 Javascript
json格式的javascript对象用法分析
2016/07/04 Javascript
HTML页面定时跳转方法解析(2种任选)
2016/12/22 Javascript
修改ligerui 默认确认按钮的方法
2016/12/27 Javascript
Vue.js系列之项目搭建(1)
2017/01/03 Javascript
webpack实现热更新(实施同步刷新)
2017/07/28 Javascript
微信小程序实现轮播图效果
2017/09/07 Javascript
微信小程序上传图片到服务器实例代码
2017/11/07 Javascript
Vue scrollBehavior 滚动行为实现后退页面显示在上次浏览的位置
2019/05/27 Javascript
JavaScript迭代器的含义及用法
2019/06/21 Javascript
使用layui 的layedit定义自己的toolbar方法
2019/09/18 Javascript
使用vue实现一个电子签名组件的示例代码
2020/01/06 Javascript
uniapp电商小程序实现订单30分钟倒计时
2020/11/01 Javascript
通过滑动翻页效果实现和移动端click事件问题
2021/01/26 Javascript
Python实现读取并保存文件的类
2017/05/11 Python
Tornado协程在python2.7如何返回值(实现方法)
2017/06/22 Python
Python中多个数组行合并及列合并的方法总结
2018/04/12 Python
Tensorflow 实现释放内存
2020/02/03 Python
CSS3 透明色 RGBA使用介绍
2013/08/06 HTML / CSS
CSS Grid布局教程之浏览器开启CSS Grid Layout汇总
2014/12/30 HTML / CSS
html5中使用hotcss.js实现手机端自适配的方法
2020/04/23 HTML / CSS
新文化运动的基本口号
2014/06/21 职场文书
领导班子自我剖析材料
2014/08/16 职场文书
化验室岗位职责
2015/02/14 职场文书
不同意离婚代理词
2015/05/23 职场文书
如何在pycharm中快捷安装pip命令(如pygame)
2021/05/31 Python
利用Python实现翻译HTML中的文本字符串
2022/06/21 Python
Win11 vmware不兼容怎么办?Win11与VMware虚拟机不兼容的解决方法
2023/01/09 数码科技