我们平时对 cookie 的增删改查等操作,都是在操作 document.cookie,这里我们介绍一个新方法 document.cookie 能获取到当前域所有的 cookie 字符串。每个 cookie 用分号进行隔开: 操作 cookie,均是在操作 document.cookie。如下面就是我常用的一段代码: 可以看到设置、获取和删除 cookie,都是在 document.cookie 上进行操作的。 现在 Chrome 有了更方便操作 cookie 的方法了 下图是当前日期 2021/03/15 的兼容性概览,可以发现仅仅是 Chrome 体系支持了 cookieStore。 不过我们可以先来了解它的用法。 cookieStore 现在只能在 cookieStore 是一个类似 可以看到 cookieStore 主要有 5 个方法: 前 4 个方法天然支持 Promise。接下来我们一个个来了解下。 cookieStore.set 方法可以设置 cookie,并返回一个 Promise 状态,表示是否设置成功。 如果我们想要设置更多的属性,例如过期时间,可以传入一个 Object 类型: value 中所有的数据都会默认先执行 上面都是我们设置 cookie 成功的情况,那么什么时候会设置失败呢?在本地 localhost 环境会设置失败。 本地 localhost,我们是能获取到 浏览器会发出提示,无法在不安全的域名下通过 CookieStore 中的 set 设置 cookie: 添加 catch 后,就能捕获到这个错误: 因此在想使用 cookieStore 时,不能直接通过下面的方式判断,还得新增一个页面 url 的协议来判断: 应当使用: get()方法还可以接收一个 Object 类型,测试后发现,key 的值只能是 name: 当获取的 cookie 不存在时,则返回一个 getAll()方法也可以传入一个 name,用来获取对应的 cookie: 删除成功后则会提示删除成功。 即使删除一个不存在的 cookie,也会提示删除成功。因此,当再次执行上面的代码时,还是会正常提示。 同样的,在 localhost 环境下会提示删除失败。 我们可以通过添加 添加监听事件: 这里面有 2 个重要的字段 当调用 set()方法时,会触发 change 事件,同时影响的 cookie 会放在 通过 document.cookie 设置或者删除的 cookie,均认为是在修改 cookie,而不是删除。 每次设置 cookie 时,即使两次的 name 和 value 完全一样,也会触发 通过 delete()方法删除一个存在的 cookie 时,会触发 change 事件,被删除的 cookie 会放在 如果删除一个不存在的 cookie,则不会触发 change 事件。 可以看到,当第二次删除 name 为 math 的 cookie 时,就没有触发 change 事件。 我们整篇了解了下cookieStore
。1. 平时如何操作 cookie #
document.cookie; // "a=1; b=2; c=wenzi"
/**
* 写cookies
* @param {string} name 写cookie的key
* @param {string|number} value 写cookie的值
* @param {number} day 存储的时间,默认30天
*/
export const setCookie = (name: string, value: string | number, day = 30): void => {
const exp = new Date();
exp.setTime(exp.getTime() + day * 24 * 60 * 60 * 1000);
document.cookie = `${name}=${escape(value.toString())};path=/;expires=${exp.toUTCString()}`;
};
/**
* 读取cookies
* @param {string} name 要获取的cookie名称
* @param {number|boolean} type 是否直接获取对应的值,若存入真值,则直接返回,否则进行解码
*/
export const getCookie = (name: string): string | null => {
const reg = new RegExp(`(^| )${name}=([^;]*)(;|$)`);
const arr = document.cookie.match(reg);
if (arr) {
return unescape(arr[2]);
}
return null;
};
/**
* 删除cookie
* @param name 删除的cookie名称
*/
export const delCookie = (name: string) => {
if (!name) return;
const ex: Date = new Date();
ex.setTime(ex.getTime() - 1);
document.cookie = `${name}=; expires=${ex.toUTCString()};path=/`;
};
2. 新方式 cookieStore #
cookieStore
,这个方法是在 Chrome87 版本加入的,兼容性还不太好。https 协议
下的域名才能访问的到;其他 http 协议的域名里会提示 cookieStore 为 undefined,或者设置失败。2.1 基本方法 #
localStorage
的 object 类型变量。
2.2 设置 cookie #
cookieStore
.set('username', 'wenzi')
.then(() => console.log('设置username成功'))
.catch(() => console.error('设置username失败'));
cookieStore
.set({
name: 'age',
value: 18,
expires: new Date().getTime() + 24 * 60 * 60 * 1000,
})
.then(() => console.log('设置age成功'))
.catch(() => console.error('设置age失败'));
toString()
,然后再进行存储,因此有些非基本类型的数据,最好先转换好。cookieStore
这个全局变量,也能执行相应的方法,但无法设置成功:cookieStore.set('username', 'wenzi');
Uncaught (in promise) TypeError: Failed to execute 'set' on 'CookieStore': Cannot modify a secure cookie on insecure origin
cookieStore
.set('username', 'wenzi')
.then(() => console.log('设置username成功'))
.catch(() => console.error('设置username失败'));
typeof cookieStore === 'object'; // 判断不准确,本地localhost也会存在
const isSupportCookieStore = typeof cookieStore === 'object' && location.protocol === 'https:'; // 只有在https协议下才使用cookieStore
2.3 获取 cookie #
cookieStore.get(name)
方法可以获取 name 对应的 cookie,会以 Promise 格式返回所有的属性:await cookieStore.get('username');
await cookieStore.get({ name: 'username' });
Promise
。2.4 获取所有的 cookie #
cookieStore.getAll()
方法可以获取当前所有的 cookie,以 Promise<[]>的形式返回的形式返回,数组中的每一项与通过 get()方式获取到的格式一样;若当前域没有 cookie,或者获取不到指定的 cookie,则为空数组;await cookieStore.getAll();
await cookieStore.getAll('username');
await cookieStore.getAll({ name: 'username' });
2.5 删除 cookie #
cookieStore.delete(name)
用来删除指定的 cookie:cookieStore
.delete('age')
.then(() => console.log('删除age成功'))
.catch(() => console.error('删除age失败'));
2.6 监听 cookie 的变化 #
change
事件,来监听 cookie 的变化,无论是通过 cookieStore 操作,还是直接操作 document.cookie,都能监听。cookieStore.addEventListener('change', (event) => {
const type = event.changed.length ? 'change' : 'delete';
const data = (event.changed.length ? event.changed : event.deleted).map((item) => item.name);
console.log(`刚才进行了 ${type} 操作,cookie有:${JSON.stringify(data)}`);
});
changed
数组和deleted
数组,当设置 cookie 时,则 changed 数组里为刚才设置的 cookie;当删除 cookie 时,则 deleted 数组里为刚才删除的 cookie。2.6.1 设置操作 #
event.changed
数组中。change
事件。cookieStore.set('math', 90);
2.6.2 删除操作 #
event.deleted
数组中。cookieStore.delete('math');
3. 总结 #
cookieStore
的使用和操作,要比我们直接操作 cookie 简便的多,而且还可以通过自身的 change 事件来监听 cookie 的变化。
版权属于:
加速器之家
作品采用:
《
署名-非商业性使用-相同方式共享 4.0 国际 (CC BY-NC-SA 4.0)
》许可协议授权
评论