javascript权威指南-第二十章(客户端存储)


layout: post
title: javascript权威指南-第二十章
date: 2017-09-07 14:36:27
tags: 读书笔记


概述

二十章讲的是客户端存储,很详细的介绍了常见的几种客户端存储的方式。涉及到一些同源策略的东西,所以就想看一下13章的安全性的问题。也记录一下

安全性

同源策略:脚本不能读取不同服务器载入的文档内容。比如说一个页面的脚本不能监听其他页面的用户输入信息,这就是同源策略。这么来看同源策略是必须的,不然浏览一个网页,所有的其他页面的信息都暴露了。
但是确实有场景需要读取其他页面的属性,可以使用document的domain属性,domain默认是载入文档的主机名,可以人为设置,比如home.example.com可以设置成example.com。不能设置成home.example或者ample.com之类的,也必须有一个.(避免出现com的情况)如果2个窗口的domain一致,就没有同源限制啦。
跨域资源共享:origin,请求头设置Access-Control-Allow-Origin

XSS(粗略了讲了一点安全问题)

对于url:http://www.example.com/greet.html?David

    let name = decodeURIComponent(window.location.search.substring(1)) // search可以拿到?之后的字段(可以理解为query字段???),转码之后可以拿到David
    alert(name)

但是如果调用http://http.www.example.com/greet.html?name=%3Cscript src=siteB/evil.js%3E%3C/script%3E
这样会注入一个来自siteB的脚本,脚本在sideB中,但是可以在A中进行操作,比较危险

防止XSS的,通过不可信的数据来的时候,去掉里面的HTML标签(包括script标签),
简单的实现安全函数

name = name.replace(/')

localStorage和sessionStorage的区别

两者的区别在于存储的有效期作用域不同,也就是可以存储多长时间以及谁对数据有访问权。
storage都定义了存储、获取、删除的方法

有效期和作用域

localStorage的作用域是文档源级别。文档源就是协议主机和端口一致,也就是说同一个浏览器开多个窗口打开一个页面,是可以共享localStorage的。但是受浏览器供应商的限制。
sessionStorage的作用域也是文档源级别,但是不太相同,它是限定在窗口中的。在不同的标签中不共享。
sessionStorage在浏览器或者标签页关闭,sessionStorage里面存的数据会被删除。

存储的API

localStorage.setItem('x', 1)
localStorage.getItem('X')
/// 遍历
for(let i= 0; i< localStorage.length; i++ ){
    let name = localStorage.key(i)
    let value = localStorage.getItem(name)
}
localStorage.removeItem('X')    //删除X项
localStorage.clear()            //全部删除

cookie

cookie属性:有效期和作用域

cookie的默认有效期是浏览器的会话期间。但是和sessionStorage有区别,cookie不局限在浏览器单个窗口,有效期和整个浏览器的进程有关。
通过max-age设置有效期设置有效期,他的作用域可以通过path和domain来配置,默认情况和创建他的web页面有关,对子页面可见。

如果路径设置成'/'就和localStorage一样的作用域了

保存cookie

// cookie中不允许包含分毫、逗号和空白符,所以一般采用encodeURIComponent转码
// 最简单的cookie设置,浏览器关闭清除
document.cookie = 'version=' + encodeURIComponent(document.lastModified)

// 一个可选max-age、path、domain、secure属性的方法
// 通过设置domain可以解决cookie跨域的问题
const setCookie = (name, value, daysTolive, path, path) = {
    let cookie = `${name}=${encodeURLComponent(value)}`
    if(typeOf daysToLive === 'number'){
        cookie += `; max-age=${daysToLive * 60 * 60 * 24}`
    }
    if(path){
        cookie += `; path=${path}`
    }
    if(domain){
        cookie += `; domain=${domain}`
    }
    document.cookie = cookie
}

读取cookie

通过document.cookie读取出来的是一个字符串,不同键值对通过f分号和空格隔开。value的解码方式取决于之前存储用的方式,例如上面用到的setCookie中用的方式是encodeURIComponent编码

const getCookie = (){
    let cookieObj = {}
    let cookieStr = document.cookie
    if(cookieStr === ''){
        return cookieObj
    }
    let cookieArr = cookieStr.split('; ')
    for(let _cookie of cookieArr){
        let p = _cookie.indexOf('=')
        let name = _cookie.slice(0,p)
        let value = _cookie.slice(p+1)
        value = decodeURIComponent(value)
        cookieObj[name] = value
    }
    return cookieObj
}

cookie相关的存储

实现类似于sessionStorage的getItem、setItem、remove、delete等功能

const cookieStorage(maxage, path) {
    //自执行函数实例化一个cookie对象
    let cookie = (()=>{
        let cookieObj = {}
        let cookieStr = document.cookie
        if(cookieStr === ''){
        return cookieObj
        }
        let cookieArr = cookieStr.split('; ')
        for(let _cookie of cookieArr){
            let p = _cookie.indexOf('=')
            let name = _cookie.slice(0,p)
            let value = _cookie.slice(p+1)
            value = decodeURIComponent(value)
            cookieObj[name] = value
        }
        return cookieObj
    }())
    // 拿出所有的key
    let keys = Object.keys(cookie)
    //存储API公共的属性和方法
    this.length = keys.length
    //返回第n个key的名字,即key方法
    this.key = (n) => {
        if (n<0 || n >= keys.length) return null
        return keys[n]
    }
    //getItem方法
    this.getItem = (name) => cookie[name] || null
    // setItem方法
    this.setItem = (key,value) => {
        //首先判断要存储的cookie存在不,不存在先操作一次keys和length
        if(!(key in keys)){
            keys.push(key)
            this.length++
        }
        // 针对cookie对象做的操作
        cookie[key] = value
        //接下来就是正式操作浏览器中的cookie
        let domCookie = `${key}=${encodeURIComponent(value)}`
        if(maxage) domCookie += `; max-age=${maxage}`
        if(path) domCookie += `; path=${path}`
        document.cookie = domCookie
    }
    // removeCookie
    this.removeItem = (key) => {
        if(!(key in cookie)) return
        delete cookie[key]
        this.length--
        let p = keys.indexOf(key)
        keys.splice(p,1)
        document.cookie = 
    }
    // clear
    this.clear = () => {
        for(let key of keys){
            document.cookie = `${key}=; max-age=0`
        }
        cookie = {}
        keys = []
        this.length = 0
    }
}

你可能感兴趣的:(javascript权威指南-第二十章(客户端存储))