偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

超越Cookie:當(dāng)今的客戶端數(shù)據(jù)存儲(chǔ)技術(shù)

數(shù)據(jù)庫(kù) 其他數(shù)據(jù)庫(kù)
當(dāng) cookie 被首次引入時(shí),它是瀏覽器保存數(shù)據(jù)的唯一方式。之后又有了很多新的選擇:Web Storage API、IndexedDB 和 Cache API。那么 cookie 死了嗎?我們來看看這些在瀏覽器中存儲(chǔ)數(shù)據(jù)的技術(shù)。

[[272939]]

當(dāng) cookie 被首次引入時(shí),它是瀏覽器保存數(shù)據(jù)的唯一方式。之后又有了很多新的選擇:Web Storage API、IndexedDB 和 Cache API。那么 cookie 死了嗎?我們來看看這些在瀏覽器中存儲(chǔ)數(shù)據(jù)的技術(shù)。

Cookies

Cookie 是由服務(wù)器發(fā)送或在客戶端上設(shè)置的信息單位,保存在用戶的本地瀏覽器上。它們會(huì)自動(dòng)附加到每個(gè)請(qǐng)求上。由于 HTTP 是無狀態(tài)協(xié)議,因此 cookie 允許將信息存儲(chǔ)在客戶端上,以便將其他上下文數(shù)據(jù)傳給該服務(wù)器。

Cookie 有一些標(biāo)志,對(duì)于提高數(shù)據(jù)的安全性非常有用。 HttpOnly 標(biāo)志阻止用 JavaScript 訪問 cookie 的行為,只有附加在 HTTP 請(qǐng)求上時(shí)才能訪問它們。這非常適合防止通過 XSS(跨站點(diǎn)腳本)攻擊造成數(shù)據(jù)泄露。

此外,Secure 標(biāo)志確保僅在通過 HTTPS 協(xié)議發(fā)送請(qǐng)求時(shí)才發(fā)送 cookie。 SameSite 標(biāo)志,可以設(shè)置為 lax 或 strict(它們的差異看這里),可用于幫助防止 CSRF(跨站點(diǎn)請(qǐng)求偽造)請(qǐng)求。它告訴瀏覽器只有在請(qǐng)求是與請(qǐng)求者在同一域中的 URL 時(shí)才發(fā)送 cookie。

什么時(shí)候使用 cookies?

那么,在哪些情況下你希望獲得 Cookie?最常見的應(yīng)用場(chǎng)景之一是授權(quán) token 。由于 HttpOnly 標(biāo)志為 XSS 攻擊添加了額外的保護(hù)層,SameSite 可以防止 CSRF,而 Secure 可以確保你的 cookie 被加密,這使你的身份驗(yàn)證token 有額外的保護(hù)層。

由于 auth token 非常小,因此你無需擔(dān)心請(qǐng)求過大。此外由于它們會(huì)自動(dòng)附加到每個(gè)請(qǐng)求,因此使用 cookie 可以在服務(wù)器上確定用戶是否經(jīng)過身份驗(yàn)證。這對(duì)于服務(wù)器呈現(xiàn)的內(nèi)容非常有用,例如你希望將未經(jīng)過身份驗(yàn)證的用戶重定向到登錄頁面。

Cookie 的另一個(gè)用途是存儲(chǔ)用戶的語言代碼。由于你可能希望在大多數(shù)請(qǐng)求中訪問用戶的語言,因此你可以利用它自動(dòng)附加。

如何使用 cookies?

前面經(jīng)討論了要使用 cookie 的原因,現(xiàn)在來看看你可以如何使用 cookie。要從服務(wù)器上給客戶端設(shè)置 cookie,需要在 HTTP 響應(yīng)中添加 Set-Cookie 標(biāo)頭。 Cookie 應(yīng)采用 key=value 的格式。如果你要在 Node.js 程序中設(shè)置 cookie,你的代碼可能像下面這樣:

 

  1. response.setHeader('Set-Cookie', ['user_lang=en-us', 'user_theme=dark_mode']); 

這將會(huì)設(shè)置兩個(gè) cookie:它將 user_lang 設(shè)置為 en-us,將 user_theme 設(shè)置為 dark_mode。

Cookie 也可以由客戶端操縱。要設(shè)置 cookie,可以用 key=value 的格式為 document.cookie 賦值。如果 key 已存在,則會(huì)被覆蓋掉。

 

  1. document.cookie = 'user_lang=es-es'

如果已經(jīng)定義了 user_lang,它現(xiàn)在等于es-es。

你可以通過訪問 document.cookie 值來查看所有的 cookie。這將返回一串以分號(hào)做分隔的鍵值對(duì)。

 

  1. document.cookie = 'user_lang=en-us' 
  2. document.cookie = 'user_theme=light_mode' 
  3. console.log(document.cookie); // 'user_lang=en-us; user_theme=light_mode;' 

要增加鍵值對(duì)的可訪問性,可以使用以下函數(shù)將此字符串解析為對(duì)象:

 

  1. const parseCookies = x => x  
  2.   .split(';')  
  3.   .map(e => e.trim().split('='))  
  4.   .reduce((obj, [key, value]) => ({...obj, [key]: value}), {}); 

If you need to set one of the flags onto your cookie, you can add them after a semicolon. For example, if you’d like to set the Secure and SameSite flags onto your cookie, you would do the following:

如果你需要將其中一個(gè)標(biāo)志設(shè)置到 cookie 上,可以在分號(hào)后添加它們。例如你想在 Cookie 上設(shè)置 Secure 和 SameSite 標(biāo)志,則可以執(zhí)行以下操作:

 

  1. document.cookie = 'product_ids=123,321;secure;samesite=lax' 

由于 HTTPOnly 的作用是使 cookie 只能在服務(wù)器上訪問,因此它只能由服務(wù)器添加。

除了這些安全標(biāo)志之外,你還可以設(shè)置 Max-Age( cookie 應(yīng)該保存的秒數(shù))或 Expires(Cookie應(yīng)該過期的日期)。如果這些都未設(shè)置,則 cookie 將跟隨瀏覽器會(huì)話的持續(xù)時(shí)間。如果用戶使用隱身模式,則會(huì)在用戶會(huì)話關(guān)閉時(shí)刪除 Cookie。

由于處理 cookie 的接口不是很友好,所以你可以使用諸如 js-cookie 之類的庫(kù)來方便對(duì)其的操作。

Web Storage API

Web Storage API 是一種在本地存儲(chǔ)數(shù)據(jù)的新選項(xiàng)。它在 HTML5 中中添加,Web Storage API 包括localStorage 和 sessionStorage。雖然 cookie 通常處理 server/client 通信,但 Web Storage API 最適用于保存客戶端數(shù)據(jù)。

我們已經(jīng)將 cookie 作為在本地存儲(chǔ)數(shù)據(jù)的選項(xiàng),為什么還需要 Web 存儲(chǔ)?其中一個(gè)原因是:由于 cookie 會(huì)自動(dòng)添加到每個(gè) HTTP 請(qǐng)求中,因此請(qǐng)求大小會(huì)變得臃腫。所以你可以用 Web Storage API 存儲(chǔ)比 cookie 更大量的數(shù)據(jù)。

另一個(gè)優(yōu)點(diǎn)是更直觀的 API。如果使用 cookie,你需要手動(dòng)解析 cookie 字符串來訪問各個(gè)鍵。 Web Storage 使這更加容易。如果要設(shè)置或獲取值,可以使用 setItem 或 getItem。

 

  1. localStorage.setItem('selected_tab', 'FAQ');  
  2. localSTorage.getItem('selected_tab'); // 'FAQ' 

鍵和值都必須是字符串。如果你想保存一個(gè)對(duì)象或數(shù)組,可以在保存時(shí)調(diào)用 JSON.stringify() 并在讀取時(shí)調(diào)用 JSON.parse() 來實(shí)現(xiàn)。

 

  1. const product = {  
  2.   id: '123',  
  3.   name: 'Coffee Beans',  
  4. };  
  5. localStorage.setItem('cached_product', JSON.stringify(product));  
  6. JSON.parse(localStorage.getItem('cached_product')); 

local storage 的另一個(gè)用例是在多個(gè)選項(xiàng)卡之間同步數(shù)據(jù)。通過為 'storage' 事件添加偵聽器,你可以在另一個(gè)選項(xiàng)卡或窗口中更新數(shù)據(jù)。

 

  1. window.addEventListener('storage', () => {  
  2.   console.log('local storage has been updated');  
  3. }); 

僅當(dāng)在另一個(gè)文檔中修改本地或會(huì)話存儲(chǔ)時(shí)才會(huì)觸發(fā)此事件。也就是說,你無法在當(dāng)前瀏覽器選項(xiàng)卡中偵聽 storage 的更改。不幸的是,截至撰寫本文時(shí),存儲(chǔ)事件監(jiān)聽器尚未在 Chrome 上得到支持。

那么localStorage 和 sessionStorage 之間有什么區(qū)別呢?與 cookie 不同,Web Storage API 沒有過期或最大期限功能。如果使用 localStorage,除非手動(dòng)刪除,否則數(shù)據(jù)將無限期保留。你可以通過運(yùn)行 localStorage.removeItem('key') 來刪除單個(gè)鍵的值,或者通過運(yùn)行 localStorage.clear() 清除所有數(shù)據(jù)。

如果使用 sessionStorage,則數(shù)據(jù)將僅持續(xù)到當(dāng)前會(huì)話結(jié)束。如果你沒有設(shè)置最大時(shí)間或過期,它將被視為與 cookie 保持的方式相似。在任何一種情況下,如果用戶使用隱身,本地存儲(chǔ)都不會(huì)在會(huì)話之間保留數(shù)據(jù)。

IndexedDB

如果 cookie 和 localStorage 都不符合你的要求,還有另一種選擇:IndexedDB,一個(gè)瀏覽器內(nèi)置的數(shù)據(jù)庫(kù)系統(tǒng)。

當(dāng) localStorage 同步執(zhí)行所有方法時(shí),IndexedDB 會(huì)異步調(diào)用它們。這將會(huì)允許訪問數(shù)據(jù)而不會(huì)阻塞其余代碼。當(dāng)你處理大量可能訪問代價(jià)高昂的代碼時(shí),這非常有用。

IndexedDB 在其存儲(chǔ)的數(shù)據(jù)類型方面也具有更大的靈活性。雖然 cookies 和 localStorage 僅限于存儲(chǔ)字符串,但 IndexedDB 可以存儲(chǔ)可以通過“結(jié)構(gòu)化克隆算法”復(fù)制的任何類型的數(shù)據(jù)。這包括 Object、 Date、 File、 Blob、 RegEx 以及更多類型。

性能和靈活性增加的缺點(diǎn)是 IndexedDB 的 API 更低級(jí)且更復(fù)雜。幸運(yùn)的是有許多庫(kù)可以解決這個(gè)問題。

localForage 為 IndexedDB 提供了一個(gè)更簡(jiǎn)單的類似 localStorage 的 API。 PouchDB 提供了一個(gè)可以離線的存儲(chǔ) API,可以與在線 CouchDB 數(shù)據(jù)庫(kù)同步。 idb 是一個(gè)小型庫(kù),具有更簡(jiǎn)單的基于 promise 的 API。 Dexie 添加了更強(qiáng)大的查詢 API,同時(shí)保持了良好的性能。根據(jù)你的使用情況還有許多選擇。

Cache API

另一種用于持久數(shù)據(jù)的專用工具是 Cache API。雖然它最初是為 service workers 創(chuàng)建的,但它可用于緩存任何網(wǎng)絡(luò)請(qǐng)求。 Cache API 公開了 Window.caches,它提供了保存和檢索響應(yīng)的方法,允許你保存可永遠(yuǎn)以后訪問的 Requests 和 Responses 對(duì)。

例如,如果你想在從 API 請(qǐng)求響應(yīng)之前檢查瀏覽器的緩存以獲取響應(yīng),則可以執(zhí)行以下操作:

 

  1. const apiRequest = new Request('https://www.example.com/items');  
  2. caches.open('exampleCache') // opens the cache  
  3.   .then(cache => {  
  4.     cache.match(apiRequest) // checks if the request is cached  
  5.       .then(cachedResponse =>   
  6.         cachedResponse || // return cachedReponse if available  
  7.         fetch(apiRequest) // otherwise, make new request  
  8.           .then(response => {  
  9.             cache.put(apiRequest, response); // cache the response  
  10.             return response;  
  11.           })  
  12.         })  
  13.     .then(res => console.log(res))  
  14. }) 

第一次運(yùn)行代碼時(shí),它將緩存響應(yīng)。隨后每次都會(huì)緩存請(qǐng)求,并且不會(huì)發(fā)出網(wǎng)絡(luò)請(qǐng)求。

總結(jié)

在瀏覽器上存儲(chǔ)數(shù)據(jù)的每種方法都有其自己的用途。如果信息很小,很敏感,并且可能在服務(wù)器上使用,那么 cookie 就是最佳選擇。如果要保存更大且更不敏感的數(shù)據(jù),Web Storage API 可能是更好的選擇。

如果你打算存儲(chǔ)大量結(jié)構(gòu)化數(shù)據(jù),IndexedDB 非常棒。 Cache API 用于存儲(chǔ)來自 HTTP 請(qǐng)求的響應(yīng)。根據(jù)你的需要,有很多工具可供使用。

責(zé)任編輯:龐桂玉 來源: segmentfault
相關(guān)推薦

2016-10-20 16:11:39

HtmlJavascript

2017-04-06 15:40:31

客戶端數(shù)據(jù)存儲(chǔ)技術(shù)

2011-05-24 16:47:20

數(shù)據(jù)存儲(chǔ)

2009-07-21 13:03:06

桌面虛擬化虛擬PC數(shù)據(jù)中心

2013-01-10 10:04:53

離線VDI客戶端hypervis

2021-09-22 15:46:29

虛擬桌面瘦客戶端胖客戶端

2015-06-03 09:27:05

JavaScript客戶端檢測(cè)技術(shù)

2011-03-21 14:53:36

Nagios監(jiān)控Linux

2011-04-06 14:24:20

Nagios監(jiān)控Linux

2011-08-17 10:10:59

2010-12-21 11:03:15

獲取客戶端證書

2011-03-24 13:00:31

配置nagios客戶端

2011-03-02 14:36:24

Filezilla客戶端

2011-10-26 13:17:05

2010-05-31 10:11:32

瘦客戶端

2020-03-12 19:00:48

Ceph分布式存儲(chǔ)

2010-07-29 09:08:20

Flex客戶端緩存

2010-07-22 12:24:31

Telnet客戶端

2011-03-25 12:50:29

nagios安裝

2010-05-26 09:26:43

Cassandra
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)