网络早期最大的问题之一是如何管理状态。简而言之,服务器没法知道两个请求是否来自同一个浏览器。当时最简单的方法是在请求时,在页面中插入一些参数,并在下一个请求中传回参数。这须要使用包含参数的隐藏的表单,或者做为URL参数的一部分传递。这两个解决方案都手动操做,容易出错。cookie出现来解决这个问题。javascript
cookie是纯文本,没有可执行代码。存储数据,当用户访问了某个网站(网页)的时候,咱们就能够经过cookie来向访问者电脑上存储数据,或者某些网站为了辨别用户身份、进行session跟踪而储存在用户本地终端上的数据(一般通过加密)css
当网页要发http请求时,浏览器会先检查是否有相应的cookie,有则自动添加在request header中的cookie字段中。这些是浏览器自动帮咱们作的,并且每一次http请求浏览器都会自动帮咱们作。这个特色很重要,由于这关系到“什么样的数据适合存储在cookie中”。html
存储在cookie中的数据,每次都会被浏览器自动放在http请求中,若是这些数据并非每一个请求都须要发给服务端的数据,浏览器这设置自动处理无疑增长了网络开销;但若是这些数据是每一个请求都须要发给服务端的数据(好比身份认证信息),浏览器这设置自动处理就大大免去了重复添加操做。因此对于那种设置“每次请求都要携带的信息(最典型的就是身份认证信息)”就特别适合放在cookie中,其余类型的数据就不适合了。前端
客户端设置java
document.cookie = '名字=值'; document.cookie = 'username=cfangxu;domain=baike.baidu.com' 而且设置了生效域
注意: 客户端能够设置cookie 的下列选项:expires、domain、path、secure(有条件:只有在https协议的网页中,客户端设置secure类型的 cookie 才能成功),但没法设置HttpOnly选项。web
服务器端设置
无论你是请求一个资源文件(如 html/js/css/图片),仍是发送一个ajax请求,服务端都会返回response。而response header中有一项叫set-cookie,是服务端专门用来设置cookie的。ajax
Set-Cookie 消息头是一个字符串,其格式以下(中括号中的部分是可选的):
Set-Cookie: value[; expires=date][; domain=domain][; path=path][; secure]
注意: 一个set-Cookie字段只能设置一个cookie,当你要想设置多个 cookie,须要添加一样多的set-Cookie字段。
服务端能够设置cookie 的全部选项:expires、domain、path、secure、HttpOnly
经过 Set-Cookie 指定的这些可选项只会在浏览器端使用,而不会被发送至服务器端。数据库
咱们经过document.cookie来获取当前网站下的cookie的时候,获得的字符串形式的值,它包含了当前网站下全部的cookie(为避免跨域脚本(xss)攻击,这个方法只能获取非 HttpOnly 类型的cookie)。它会把全部的cookie经过一个分号+空格的形式串联起来,例如username=chenfangxu; job=coding
后端
要想修改一个cookie,只须要从新赋值就行,旧的值会被新的值覆盖。但要注意一点,在设置新cookie时,path/domain这几个选项必定要旧cookie 保持同样。不然不会修改旧值,而是添加了一个新的 cookie。跨域
把要删除的cookie的过时时间设置成已过去的时间,path/domain/这几个选项必定要旧cookie 保持同样。
若是只设置一个值,那么算cookie中的value; 设置的两个cookie,key值若是设置的相同,下面的也会把上面的覆盖。
若是咱们想长时间存放一个cookie。须要在设置这个cookie的时候同时给他设置一个过时的时间。若是不设置,cookie默认是临时存储的,当浏览器关闭进程的时候自动销毁
注意:document.cookie = '名称=值;expires=' + GMT(格林威治时间)格式的日期型字符串;
通常设置天数:new Date().setDate( oDate.getDate() + 5 ); 比当前时间多5天
一个设置cookie时效性的例子
function setCookie(c_name, value, expiredays){ var exdate=new Date(); exdate.setDate(exdate.getDate() + expiredays); document.cookie=c_name+ "=" + escape(value) + ((expiredays==null) ? "" : ";expires="+exdate.toGMTString()) } 使用方法:setCookie('username','cfangxu',30)
expires 是 http/1.0协议中的选项,在新的http/1.1协议中expires已经由 max-age 选项代替,二者的做用都是限制cookie 的有效时间。expires的值是一个时间点(cookie失效时刻= expires),而max-age 的值是一个以秒为单位时间段(cookie失效时刻= 建立时刻+ max-age)。
另外,max-age 的默认值是 -1(即有效期为 session );max-age有三种可能值:负数、0、正数。
负数:有效期session;
0:删除cookie;
正数:有效期为建立时刻+ max-age
domain指定了 cookie 将要被发送至哪一个或哪些域中。默认状况下,domain 会被设置为建立该 cookie 的页面所在的域名,因此当给相同域名发送请求时该 cookie 会被发送至服务器。
浏览器会把 domain 的值与请求的域名作一个尾部比较(即从字符串的尾部开始比较),并将匹配的 cookie 发送至服务器。
document.cookie = "username=cfangxu;path=/;domain=qq.com"
如上:“www.qq.com" 与 "sports.qq.com" 公用一个关联的域名"qq.com",咱们若是想让 "sports.qq.com" 下的cookie被 "www.qq.com" 访问,咱们就须要用到 cookie 的domain属性,而且须要把path属性设置为 "/"。
Set-Cookie: username=cfangxu;path=/;domain=qq.com
注:必定的是同域之间的访问,不能把domain的值设置成非主域的域名。
cookie 通常都是因为用户访问页面而被建立的,但是并非只有在建立 cookie 的页面才能够访问这个 cookie。
由于安全方面的考虑,默认状况下,只有与建立 cookie 的页面在同一个目录或子目录下的网页才能够访问。
即path属性能够为服务器特定文档指定cookie,这个属性设置的url且带有这个前缀的url路径都是有效的。
最经常使用的例子就是让 cookie 在根目录下,这样无论是哪一个子页面建立的 cookie,全部的页面均可以访问到了。
document.cookie = "username=cfangxu; path=/"
Set-Cookie:name=cfangxu; path=/blog
如上设置:path 选项值会与 /blog,/blogrool 等等相匹配;任何以 /blog 开头的选项都是合法的。须要注意的是,只有在 domain 选项核实完毕以后才会对 path 属性进行比较。path 属性的默认值是发送 Set-Cookie 消息头所对应的 URL 中的 path 部分。
domain是域名,path是路径,二者加起来就构成了 URL,domain和path一块儿来限制 cookie 能被哪些 URL 访问。
因此domain和path2个选项共同决定了cookie什么时候被浏览器自动添加到请求头部中发送出去。若是没有设置这两个选项,则会使用默认值。domain的默认值为设置该cookie的网页所在的域名,path默认值为设置该cookie的网页所在的目录。
一般 cookie 信息都是使用HTTP链接传递数据,这种传递方式很容易被查看,因此 cookie 存储的信息容易被窃取。假如 cookie 中所传递的内容比较重要,那么就要求使用加密的数据传输。
secure选项用来设置cookie只在确保安全的请求中才会发送。当请求是HTTPS或者其余安全协议时,包含 secure 选项的 cookie 才能被发送至服务器。
document.cookie = "username=cfangxu; secure"
把cookie设置为secure,只保证 cookie 与服务器之间的数据传输过程加密,而保存在本地的 cookie文件并不加密。就算设置了secure 属性也并不表明他人不能看到你机器本地保存的 cookie 信息。机密且敏感的信息毫不应该在 cookie 中存储或传输,由于 cookie 的整个机制本来都是不安全的
注意:若是想在客户端即网页中经过 js 去设置secure类型的 cookie,必须保证网页是https协议的。在http协议的网页中是没法设置secure类型cookie的。
这个选项用来设置cookie是否能经过 js 去访问。默认状况下,cookie不会带httpOnly选项(即为空),因此默认状况下,客户端是能够经过js代码去访问(包括读取、修改、删除等)这个cookie的。当cookie带httpOnly选项时,客户端则没法经过js代码去访问(包括读取、修改、删除等)这个cookie。
在客户端是不能经过js代码去设置一个httpOnly类型的cookie的,这种类型的cookie只能经过服务端来设置。
cookie实际上是个字符串,但这个字符串中等号、分号、空格被当作了特殊符号。因此当cookie的 key 和 value 中含有这3个特殊字符时,须要对其进行额外编码,通常会用escape进行编码,读取时用unescape进行解码;固然也能够用encodeURIComponent/decodeURIComponent或者encodeURI/decodeURI
一般cookie的域和浏览器地址的域匹配,这被称为第一方cookie。那么第三方cookie就是cookie的域和地址栏中的域不匹配,这种cookie一般被用在第三方广告网站。为了跟踪用户的浏览记录,而且根据收集的用户的浏览习惯,给用户推送相关的广告。
cookie推荐资源
HTML5新方法,不过IE8及以上浏览器都兼容。
localStorage.setItem('username','cfangxu');
localStorage.getItem('username')
也能够获取键名 localStorage.key(0) #获取第一个键名
localStorage.removeItem('username')
也能够一次性清除全部存储 localStorage.clear()
当storage发生改变的时候触发。
注意: 当前页面对storage的操做会触发其余页面的storage事件
事件的回调函数中有一个参数event,是一个StorageEvent对象,提供了一些实用的属性,以下表:
Property | Type | Description |
---|---|---|
key | String | The named key that was added, removed, or moddified |
oldValue | Any | The previous value(now overwritten), or null if a new item was added |
newValue | Any | The new value, or null if an item was added |
url/uri | String | The page that called the method that triggered this change |
其实跟localStorage差很少,也是本地存储,会话本地存储
localStorage、sessionStorage
localStorage只要在相同的协议、相同的主机名、相同的端口下,就能读取/修改到同一份localStorage数据。
sessionStorage比localStorage更严苛一点,除了协议、主机名、端口外,还要求在同一窗口(也就是浏览器的标签页)下。
localStorage是永久存储,除非手动删除。
sessionStorage当会话结束(当前页面关闭的时候,自动销毁)
cookie的数据会在每一次发送http请求的时候,同时发送给服务器而localStorage、sessionStorage不会。
先说个会被取代的,为何会被取代,主要有如下几个缘由:
Web SQL database
草案,并且是在2010年年末,规范不支持了,浏览器厂商已经支持的就支持了,没有支持的也不打算支持了,好比IE和Firefox。Web SQL database
本质上是一个关系型数据库,后端可能熟悉,可是前端就有不少不熟悉了,虽然SQL的简单操做不难,可是也得须要学习。来自MDN的解释: indexedDB 是一种低级API,用于客户端存储大量结构化数据(包括, 文件/ blobs)。该API使用索引来实现对该数据的高性能搜索。虽然 Web Storage 对于存储较少许的数据颇有用,但对于存储更大量的结构化数据来讲,这种方法不太有用。IndexedDB提供了一个解决方案。
因此,IndexedDB
API是强大的,但对于简单的状况可能看起来太复杂了,因此要看你的业务场景来选择究竟是用仍是不用。
indexedDB
是一个基于JavaScript的面向对象的数据库。 IndexedDB
容许你存储和检索用键索引的对象;
IndexedDB 鼓励使用的基本模式以下所示:
语法: window.indexedDB.open(dbName, version)
var db; // 打开数据库,open还有第二个参数版本号 var request = window.indexedDB.open('myTestDatabase'); // 数据库打开成功后 request.onsuccess = function (event) { // 存储数据结果,后面全部的数据库操做都离不开它。 db = request.result; } request.onerror = function (event) { alert("Why didn't you allow my web app to use IndexedDB?!"); } // 数据库首次建立版本,或者window.indexedDB.open传递的新版本(版本数值要比如今的高) request.onupgradeneeded = function (event) { }
onupgradeneeded事件: 更新数据库的 schema,也就是建立或者删除对象存储空间,这个事件将会做为一个容许你处理对象存储空间的 versionchange
事务的一部分被调用。在数据库第一次被打开时或者当指定的版本号高于当前被持久化的数据库的版本号时,这个 versionchange
事务将被建立。onupgradeneeded
是咱们惟一能够修改数据库结构的地方。在这里面,咱们能够建立和删除对象存储空间以及构建和删除索引。
IndexedDB 使用对象存储空间而不是表,而且一个单独的数据库能够包含任意数量的对象存储空间。每当一个值被存储进一个对象存储空间时,它会被和一个键相关联。
// 数据库首次建立版本,或者window.indexedDB.open传递的新版本(版本数值要比如今的高) request.onupgradeneeded = function (event) { //以前我们不是在success中获得了db了么,为何还要在这获取, //由于在当前事件函数执行后才会去执行success事件 var db = event.target.result; // 建立一个对象存储空间,keyPath是id,keyGenerator是自增的 var objectStore = db.createObjectStore('testItem',{keyPath: 'id',autoIncrement: true}); // 建立一个索引来经过id搜索,id是自增的,不会有重复,因此能够用惟一索引 objectStore.createIndex('id','id',{unique: true}) objectStore.createIndex('name','name'); objectStore.createIndex('age','age'); //添加一条信息道数据库中 objectStore.add({name: 'cfangxu', age: '27'}); }
注意: 执行完后,在调试工具栏Application的indexedDB中也看不到,你得右键刷新一下。
建立索引的语法:
objectStore.createIndex(indexName, keyPath, objectParameters)
indexName:建立的索引名称,可使用空名称做为索引。 keyPath:索引使用的关键路径,可使用空的keyPath, 或者keyPath传为数组keyPath也是能够的。 objectParameters:可选参数。经常使用参数之一是unique,表示该字段值是否惟一,不能重复。例如,本demo中id是不能重复的,因而有设置:
上面的代码建好了字段,而且添加了一条数据,可是咱们若是想在onupgradeneeded事件外面操做,接下来的步骤了。
因为数据库的操做都是基于事务(transaction)来进行,因而,不管是添加编辑仍是删除数据库,咱们都要先创建一个事务(transaction),而后才能继续下面的操做。
语法: var transaction = db.transaction(dbName, "readwrite");
第一个参数是事务但愿跨越的对象存储空间的列表,能够是数组或者字符串。若是你但愿事务可以跨越全部的对象存储空间你能够传入一个空数组。若是你没有为第二个参数指定任何内容,你获得的是只读事务。由于这里咱们是想要写入因此咱们须要传入 "readwrite" 标识。
var timer = setInterval(function () { if(db) { clearInterval(timer); // 新建一个事务 var transaction = db.transaction(['testItem'], 'readwrite'); // 打开一个存储对象 var objectStore = transaction.objectStore('testItem'); // 添加数据到对象中 objectStore.add({ name: 'xiaoming', age: '12' }); objectStore.add({ name: 'xiaolong', age: '20' }); } },100)
为何要用一个间隔定时器? 由于这是一个demo,正常的是要有操做才能进行数据库的写入,在咱们的demo中,js执行到transaction会比indexedDB的onsuccess事件回调快,致使会拿到db为undefined,因此写了个间隔定时器等它一会。
var transaction = db.transaction(['testItem'], 'readwrite'); var objectStore = transaction.objectStore('testItem'); var getRquest = objectStore.get(1); getRquest.onsuccess = function (event) { console.log(getRquest.result); } //输出:{name: "cfangxu", age: "27", id: 1}
var transaction = db.transaction(['testItem'], 'readwrite'); var objectStore = transaction.objectStore('testItem'); var getRquest = objectStore.put({ name: 'chenfangxu', age: '27', id:1 }); // 修改了id为1的那条数据
var transaction = db.transaction(['testItem'], 'readwrite'); var objectStore = transaction.objectStore('testItem'); var getRquest = objectStore.delete(1); // 删除了id为1的那条数据
上面的例子执行完后,必定必定要右键刷新indexedDB,它本身是不会变的。
我有一个前端学习交流QQ群:328058344 若是你在学习前端的过程当中遇到什么问题,欢迎来个人QQ群提问,群里天天还会更新一些学习资源。禁止闲聊,非喜勿进。