浏览器缓存笔记

一、浏览器缓存中常见HTTP头部简介

1、Cache-Control响应头

public:表明响应可以被任何对象(包括:发送请求的客户端,代理服务器,等等)缓存,即使是通常不可缓存的内容。(例如:1.该响应没有max-age指令或Expires消息头;2. 该响应对应的请求方法是 POST 。)

private:表明响应只能被单个用户缓存,不能作为共享缓存(即代理服务器不能缓存它)。私有缓存可以缓存响应内容,比如:对应用户的本地浏览器。

no-cache:协商缓存,在发布缓存副本之前,强制要求缓存把请求提交给原始服务器进行验证。

no-store:不使用任何缓存,缓存不应存储有关客户端请求或服务器响应的任何内容。

max-age=<seconds>:强制缓存,设置缓存存储的最大周期,超过这个时间缓存被认为过期 (单位秒)。与Expires相反,时间是相对于请求的时间。

s-maxage=<seconds>:强制缓存,覆盖max-age或者Expires头,但是仅适用于共享缓存 (比如各个代理),私有缓存会忽略它。

2、Expect响应头

Expires: <http-date>:响应头包含日期/时间,即在此时候之后,响应过期。如果在Cache-Control响应头设置了 "max-age" 或者 "s-max-age" 指令,那么 Expires 头会被忽略。

3、ETag响应头

HTTP 响应头是资源的特定版本的标识符。这可以让缓存更高效,并节省带宽,因为如果内容没有改变,Web 服务器不需要发送完整的响应。而如果内容发生了变化,使用 ETag 有助于防止资源的同时更新相互覆盖(“空中碰撞”)。

4、Last-Modified响应头

是一个响应首部,其中包含源头服务器认定的资源做出修改的日期及时间。它通常被用作一个验证器来判断接收到的或者存储的资源是否彼此一致。由于精确度比 ETag 要低,所以这是一个备用机制。包含有 If-Modified-Since 或 If-Unmodified-Since 首部的条件请求会使用这个字段。

5、If-Modified-Since 请求头

If-Modified-Since 是一个条件式请求首部,服务器只在所请求的资源在给定的日期时间之后对内容进行过修改的情况下才会将资源返回,状态码为 200 。如果请求的资源从那时起未经修改,那么返回一个不带有消息主体的 304 响应,而在 Last-Modified 首部中会带有上次修改时间。不同于 If-Unmodified-Since, If-Modified-Since 只可以用在 GET 或 HEAD 请求中。

6、If-Unmodified-Since请求头

If-Unmodified-Since 是一个条件式请求首部,只有当资源在指定的时间之后没有进行过修改的情况下,服务器才会返回请求的资源,或是接受 POST 或其他 non-safe 方法的请求。如果所请求的资源在指定的时间之后发生了修改,那么会返回 412 (Precondition Failed) 错误。

7、If-None-Match请求头

If-None-Match 是一个条件式请求首部。对于 GET 和 HEAD 请求方法来说,当且仅当服务器上没有任何资源的 ETag 属性值与这个首部中列出的相匹配的时候,服务器端才会返回所请求的资源,响应码为 200 。如存在服务器端返回响应码 304(Not Modified,未改变)。

二、常见的缓存方式

1、强缓存

        根据 Cache-Control:max-age=<seconds>或 Expires: <http-date>判断为强制缓存。
        例如:Cache-Control 为 max-age = 3000,意思是说在 3000 秒后该资源过期。

2、协商缓存

        根据Cache-Control 值: no-cache 或 max-age=0 判断为协商缓存。

        浏览器会携带缓存标识 If-None-Match、if-modified-since 向服务器发送请求,服务器根据缓存标识来决定该资源是否过期。先判断Etag(如有) = If-None-Match 继续判断 Last-Modified(如有) = if-modified-since 如果都相等304,否则返回服务器资源和200。 
        例如:Cache-Control : no-cache  意思是使用的协商混存。

3、启发式缓存

        当不设置Cache-Control或 Expires时,触发启发式缓存。如Date创建时间大于Last-Modified一年,那么浏览器会默认进行读取本地缓存时间为1year * 0.1约为36.5天,即在接下来的36.5天里,不会发送HTTP请求,直接拉取本地缓存,表现为disk cache或者memory cach。

三、缓存完整流程

1、用户通过浏览器发起请求 。

2、根据 Cache-Control或 Expires字段判断内存中是否存在有效期的强缓存或者启发式缓存,如存在内存中返回资源,不存在继续判断。

3、根据 Cache-Control或 Expires字段判断磁盘中是否存在有效期的强缓存或者启发式缓存,如存在磁盘中返回资源,不存在继续判断。

4、根据 Cache-Control或 Expires字段判断是否是协商缓存和启发式缓存,如是按照协商缓存处理。

5、如Cache-Control: no-store ,则不进行缓存,直接从服务器拉取资源。


四、其他

浏览器和CDN缓存相互独立,CND节点收到协商缓存验证时,直接通过cnd节点中的资源进行验证。(未证实)

Cache-Control等字段经过多个nginx时,如都设置了Cache-Control值将全部体现在返回包中,不会替换。
 

<think>我们将在前端使用浏览器的本地存储(localStorage或sessionStorage)来缓存数据。同时,我们将设计一个缓存策略,包括缓存数据的存储、读取、更新和过期机制。设计思路:1.使用localStorage进行持久化存储,或使用sessionStorage在会话期间存储。2.为每个缓存的数据设置一个过期时间(例如,5分钟)。3.在读取缓存时检查是否过期,如果过期则重新获取数据并更新缓存。4.提供一个缓存管理器类,封装缓存操作。我们将创建一个名为CacheManager的类,包含以下方法:-set(key, data, ttl):存储数据,ttl为缓存有效时间(毫秒)-get(key):获取数据,如果数据不存在或已过期,则返回null-remove(key):移除缓存- clear():清除所有缓存同时,我们还可以考虑使用一个前缀(prefix)来避免与其他应用缓存冲突。代码实现如下:</think># 前端浏览器缓存数据实现方案 在前端开发中,浏览器缓存是提升应用性能和用户体验的关键技术。我将介绍几种常用的浏览器缓存方案,并提供代码实现。 ## 1. localStorage 实现方案 localStorage 提供了一种简单的键值对存储方式,数据在浏览器关闭后仍然保留。 ```javascript // 存储数据 function saveToLocalStorage(key, value) { try { const serializedValue = JSON.stringify(value); localStorage.setItem(key, serializedValue); return true; } catch (error) { console.error('保存到localStorage失败:', error); return false; } } // 读取数据 function getFromLocalStorage(key) { try { const serializedValue = localStorage.getItem(key); return serializedValue ? JSON.parse(serializedValue) : null; } catch (error) { console.error('从localStorage读取失败:', error); return null; } } // 删除数据 function removeFromLocalStorage(key) { localStorage.removeItem(key); } // 使用示例 const userSettings = { theme: 'dark', language: 'zh-CN', fontSize: 16 }; // 存储数据 saveToLocalStorage('user_settings', userSettings); // 读取数据 const savedSettings = getFromLocalStorage('user_settings'); console.log(savedSettings); // {theme: 'dark', language: 'zh-CN', fontSize: 16} // 删除数据 removeFromLocalStorage('user_settings'); ``` ### 优点: - 简单易用 - 存储容量较大(通常5-10MB) - 持久化存储(除非用户清除浏览器数据) ### 缺点: - 同步操作,可能阻塞主线程 - 只能存储字符串 - 同源策略限制 ## 2. sessionStorage 实现方案 sessionStorage 与 localStorage 类似,但数据仅在当前会话中有效。 ```javascript // 存储数据 function saveToSessionStorage(key, value) { try { const serializedValue = JSON.stringify(value); sessionStorage.setItem(key, serializedValue); return true; } catch (error) { console.error('保存到sessionStorage失败:', error); return false; } } // 读取数据 function getFromSessionStorage(key) { try { const serializedValue = sessionStorage.getItem(key); return serializedValue ? JSON.parse(serializedValue) : null; } catch (error) { console.error('从sessionStorage读取失败:', error); return null; } } // 使用示例 const sessionData = { cartItems: [1, 2, 3], lastVisitedPage: '/products' }; saveToSessionStorage('shopping_session', sessionData); ``` ### 适用场景: - 临时会话数据 - 表单草稿保存 - 页面间临时传递数据 ## 3. IndexedDB 实现方案 对于更复杂的数据结构和大数据量,IndexedDB 提供了类似数据库的存储能力。 ```javascript // 打开或创建数据库 function openDB(dbName, version) { return new Promise((resolve, reject) => { const request = indexedDB.open(dbName, version); request.onerror = (event) => { reject(`数据库打开失败: ${event.target.error}`); }; request.onsuccess = (event) => { resolve(event.target.result); }; request.onupgradeneeded = (event) => { const db = event.target.result; // 创建对象存储空间(类似表) if (!db.objectStoreNames.contains('cache')) { const store = db.createObjectStore('cache', { keyPath: 'id' }); // 创建索引 store.createIndex('expires', 'expires', { unique: false }); } }; }); } // 存储数据 async function saveToIndexedDB(data) { const db = await openDB('myCacheDB', 1); return new Promise((resolve, reject) => { const transaction = db.transaction(['cache'], 'readwrite'); const store = transaction.objectStore('cache'); // 添加过期时间 const cacheItem = { ...data, expires: Date.now() + 3600000 // 1小时后过期 }; const request = store.put(cacheItem); request.onsuccess = () => resolve(); request.onerror = (event) => reject(event.target.error); }); } // 读取数据 async function getFromIndexedDB(id) { const db = await openDB('myCacheDB', 1); return new Promise((resolve, reject) => { const transaction = db.transaction(['cache'], 'readonly'); const store = transaction.objectStore('cache'); const request = store.get(id); request.onsuccess = (event) => { const data = event.target.result; // 检查是否过期 if (data && data.expires > Date.now()) { resolve(data); } else { resolve(null); } }; request.onerror = (event) => reject(event.target.error); }); } // 使用示例 const productData = { id: 'product_123', name: '高性能笔记本电脑', price: 1299, description: '最新款高性能笔记本电脑...' }; // 存储 await saveToIndexedDB(productData); // 读取 const cachedProduct = await getFromIndexedDB('product_123'); console.log(cachedProduct); ``` ### 优点: - 支持复杂数据结构 - 异步操作,不阻塞主线程 - 存储容量大(通常50%磁盘空间) - 支持事务和索引 ### 缺点: - API 较复杂 - 需要处理数据库版本迁移 ## 4. 缓存策略实现 结合上述存储方案,实现一个带过期机制的缓存管理器: ```javascript class CacheManager { constructor(storageType = 'localStorage', prefix = 'cache_') { this.storage = storageType === 'localStorage' ? localStorage : sessionStorage; this.prefix = prefix; } // 生成带前缀的键名 getKey(key) { return `${this.prefix}${key}`; } // 设置缓存 set(key, value, ttl = 3600000) { // 默认1小时 const cacheItem = { data: value, expires: Date.now() + ttl }; try { this.storage.setItem(this.getKey(key), JSON.stringify(cacheItem)); return true; } catch (error) { console.error('缓存设置失败:', error); return false; } } // 获取缓存 get(key) { const cached = this.storage.getItem(this.getKey(key)); if (!cached) return null; try { const cacheItem = JSON.parse(cached); // 检查是否过期 if (cacheItem.expires > Date.now()) { return cacheItem.data; } else { // 过期则删除 this.remove(key); return null; } } catch (error) { console.error('缓存解析失败:', error); this.remove(key); return null; } } // 删除缓存 remove(key) { this.storage.removeItem(this.getKey(key)); } // 清除所有带前缀的缓存 clear() { Object.keys(this.storage).forEach(key => { if (key.startsWith(this.prefix)) { this.storage.removeItem(key); } }); } } // 使用示例 const cache = new CacheManager(); // 设置缓存(有效期10秒) cache.set('user_profile', { name: '张三', age: 30 }, 10000); // 立即获取 console.log(cache.get('user_profile')); // {name: '张三', age: 30} // 10秒后获取 setTimeout(() => { console.log(cache.get('user_profile')); // null (已过期) }, 10000); ``` ## 5. Service Worker 缓存策略 对于更高级的缓存需求,可以使用 Service Worker 实现离线缓存: ```javascript // service-worker.js const CACHE_NAME = 'my-app-cache-v1'; const urlsToCache = [ '/', '/index.html', '/styles/main.css', '/scripts/main.js', '/images/logo.png' ]; // 安装阶段:缓存核心资源 self.addEventListener('install', event => { event.waitUntil( caches.open(CACHE_NAME) .then(cache => cache.addAll(urlsToCache)) ); }); // 激活阶段:清理旧缓存 self.addEventListener('activate', event => { event.waitUntil( caches.keys().then(cacheNames => { return Promise.all( cacheNames.map(cacheName => { if (cacheName !== CACHE_NAME) { return caches.delete(cacheName); } }) ); }) ); }); // 请求拦截:缓存优先策略 self.addEventListener('fetch', event => { event.respondWith( caches.match(event.request) .then(response => { // 返回缓存或网络请求 return response || fetch(event.request); }) ); }); // 注册 Service Worker if ('serviceWorker' in navigator) { window.addEventListener('load', () => { navigator.serviceWorker.register('/service-worker.js') .then(registration => { console.log('ServiceWorker 注册成功:', registration.scope); }) .catch(error => { console.log('ServiceWorker 注册失败:', error); }); }); } ``` ## 缓存策略选择指南 | 方案 | 适用场景 | 容量限制 | 持久性 | 复杂度 | |------|----------|----------|--------|--------| | localStorage | 用户偏好设置、小量数据 | 5-10MB | 持久 | 低 | | sessionStorage | 会话数据、表单草稿 | 5-10MB | 会话级 | 低 | | IndexedDB | 复杂数据、大数据量 | 50%磁盘 | 持久 | 中高 | | Service Worker | 静态资源、离线应用 | 50%磁盘 | 持久 | 高 | | Cookies | 用户认证、小量数据 | 4KB | 可设置 | 低 | ## 最佳实践 1. **数据分类存储**:根据数据类型选择合适存储方案 2. **设置过期时间**:避免使用过期数据 3. **缓存版本控制**:在键名中包含版本号(如 `cache_v1_user`) 4. **错误处理**:所有缓存操作都应包裹在 try-catch 中 5. **清理机制**:定期清理过期缓存 6. **容量检查**:存储前检查可用空间 ```javascript // 检查 localStorage 可用空间 function checkLocalStorageSpace() { try { const testKey = 'storage_test'; let data = ''; // 填充数据直到达到限制 while (true) { data += '0'.repeat(1024 * 1024); // 每次增加1MB localStorage.setItem(testKey, data); } } catch (error) { localStorage.removeItem(testKey); const usedSpace = Math.round(JSON.stringify(localStorage).length / 1024); console.log(`已使用空间: ${usedSpace}KB`); } } // 使用前检查 if (checkLocalStorageSpace() > 4500) { // 当使用超过4.5MB时 clearExpiredCache(); } ``` ## 相关问题
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值