当前位置: 首页 > article >正文

【浏览器】缓存

为什么有缓存?

在一个C/S结构中,最基本的缓存分为两种:

  • 客户端缓存
  • 服务器缓存

以下重点讲客户端缓存

所谓客户端缓存,顾名思义,是将某一次的响应结果保存在客户端(比如浏览器)中,而后续的请求仅需要从缓存中读取即可,极大的降低了服务器的处理压力。

来自服务器的缓存指令

当客户端发出一个get请求到服务器,服务器希望客户端将相应资源缓存起来,服务器在响应头中加入了以下内容:

Cache-Control:max-age=3600
ETag:W/"121-171ca289ebf"
Date:Thu,30 Apr 2020 12:39:56 GMT
Last-Modified:Thu, 30Apr2o20 08:16:31GMT

这个响应头表达了下面的信息:

  • Cache-Control:max-age=3600, 缓存时间是3600秒(1小时)
  • ETag:W/“121-171ca289ebf”, 这个资源的编号是W/“121-171ca289ebf”
  • Date:Thu,30Apr202012:39:56GMT, 我给你响应这个资源的服务器时间是格林威治时间2020-04-3012:39:56
  • Last-Modified:Thu,30Apr2020g8:16:31GMT, 这个资源的上一次修改时间是格林威治时间2020-04-3088:16:31

如果客户端是其他应用程序,可能并不会理会服务器的愿望,也就是说,可能根本不会缓存任何东西。

如果刚好是一个浏览器,那么就会执行缓存:

  • 浏览器把这次请求得到的响应体缓存到本地文件中
  • 浏览器标记这次请求的请求方法和请求路径
  • 浏览器标记这次缓存的时间是3600秒
  • 浏览器记录服务器的响应时间是格林威治时间2020-04-3012:39:56
  • 浏览器记录服务器给予的资源编号W/“121-171ca289ebf”
  • 浏览器记录资源的上一次修改时间是格林威治时间2020-04-3088:16:31

来自客户端的缓存协议

当客户端准备再次请求GET /index.js时,它突然想起了一件事:我需要的东西在不在缓存里呢?
此时,客户端会到缓存中去寻找是否有缓存的资源,寻找的过程如下:

  1. 缓存中是否有匹配的请求方法和路径?
  2. 如果有,该缓存资源是否有效呢?

以上两个验证会导致浏览器产生不同的行为

image.png

缓存无效(协商缓存)

如果缓存有效则直接使用缓存结果。当浏览器发现缓存已经过期,它并不会简单的把缓存删除,而是抱着一丝希望,想问问服务器,我这个缓存还能继续使用吗?

于是,浏览器向服务器发出了一个带缓存的请求,又称之为协商缓存。

所谓带缓存的请求,无非就是加入了以下的请求头:

If-Modified-Since: Thu,30 Apr 2020 08:16:31 GMT
If-None-Match: W/"121-171ca289ebf"

它们表达了下面的信息:

  • If-Modified-Since:Thu,30Apr202088:16:31GMT 这个资源
    的上一次修改时间是格林威治时间2020-04-3088:16:31,请问这个资源在这个时间之后有
    发生变动吗?
  • If-None-Match:W/“121-171ca289ebf” 这个资源的编号是W/"121-171ca289ebf, 请问这个资源的编号发生变动了吗?

其实,这两个问题可以合并为一个问题:就是问服务器资源到底变化了没有!

之所以要发两个信息,是为了兼容不同的服务器,因为有些服务器只认If-Modified-Since,有些服务器只认If-None-Match(优先级更高),有些服务器两个都认。

服务器可能会产生两个情况:

  • 缓存已经失效
  • 缓存仍然有效

如果是第一种情况一一缓存已经失效,那么非常简单,服务器再次给予一个正常的响应(响应码 200 带响应体),同时可以附带上新的缓存指令,这就回到了上一节一一来自服务器的缓存指令。

这样一来,客户端就会重新缓存新的内容。

但如果服务器觉得缓存仍然有效,它可以通过一种极其简单的方式告诉客户端:

  • 响应码为 304 Not Modified (未修改)
  • 无响应体
  • 响应头带上新的缓存指令,见上一节一一来自服务器的缓存指令

这样一来,就相当于告诉客户端:「你的缓存资源仍然可用,我给你一个新的缓存时间,你那边更新一下就可以了」

于是,客户端就继续使用缓存了。

这样一来,可以最大程度的减少网络传输,因为如果资源还有效,服务器就不会传输消息体。

强缓存和协商缓存

http 缓存分为强缓存和协商缓存。他们用于优化网站性能并减少服务器负载。这两种缓存都通过 HTTP 响应头来控制,它们分别基于不同的缓存验证方式,可以根据资源的特性和需求来选择合适的缓存策略。

客户端

<html>
<head>
    <meta charset="UTF-8">
    <meta name="viewport"
          content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
</head>
<body>
<button id="btn">send</button>
<script>
const btn = document.querySelector('#btn')
btn.addEventListener('click', ()=> {
    fetch('http://localhost:3000/api3')
})
</script>
</body>
</html>

在这里插入图片描述

在这里插入图片描述

1. 强缓存(Strong Cache)

  • 客户端在请求资源时,会检查缓存的相关响应头(如使用Cache-ControlExpires来控制缓存在本地的有效期),如果资源的缓存尚未过期,客户端将直接从本地缓存中获取资源,而不会发送请求到服务器。
  • 常用的强缓存响应头包括:
    • Cache-Control: max-age=<seconds>:指定资源的缓存时间,单位为秒。HTTP1.1 提出,优先级更高,请求头和响应头都支持这个属性,通过它提供的不同的值来定义缓存策略。此外,Cache-Control 还支持其他的属性字段,如 no-cache 、no-store、public、private 等。
    • Expires: <date>:指定资源的过期时间,是一个日期字符串,表示绝对时间,由服务器返回。它是一个 HTTP 1.0 的头部字段。受限于本地时间,修改了本地时间,可能会导致缓存失效。如果在Expires之内,则浏览器会直接读取缓存,不再请求服务器。
  • 从浏览器读取缓存分为内存缓存(memory cache,浏览器内存,关闭页面后会被清除,通常用于在同一个页面会话中的快速重复请求)和硬盘缓存(disk cache,计算机硬盘,空间大,读取效率低,持久化存储资源,直到缓存策略决定缓存到期或被手动清除),而这两种缓存策略由浏览器自身分配。(此外, 还有 Service Worker 缓存:应用程序控制的缓存层,通过 Service Worker 脚本精确控制缓存和资源请求逻辑)
  • 状态码为 200
import express from 'express'
import cors from 'cors'
import fs from 'node:fs'
import crypto from 'node:crypto'

const app = express()
app.use(cors())

//静态资源缓存 html, css, js, png
// app.use(express.static('./static', {
//     maxAge: 100 * 60 * 5,
//     lastModified: true
// }))

//动态资源缓存
//Expires
app.get('/api', (req, res) => {
    // 到了过期时间之后将不再进行缓存
    res.setHeader('Expires', new Date('2024-4-6 8:57:00').toUTCString())
    res.send('Expires1111')
})
// Cache-Control
// public 任何服服务器都可以缓存包括代理服务器 cdn
// private 只能浏览器缓存 不包括代理服务器
// max-age 缓存的时间
app.get('/api2', (req, res) => {
    res.setHeader('Cache-Control', 'public, max-age=10')
    res.send('Cache-Control1111')
})

app.listen(3000, () => {
    console.log('3000端口已启用')
})

2. 协商缓存(Conditional Cache)

当浏览器对某个资源的请求没有命中强缓存,就会发一个请求(包含If-Modified-SinceIf-None-Match的请求头)到服务器,验证协商缓存是否命中,如果协商缓存命中,请求响应返回的HTTP状态为304 (Not Modified),该请求不携带实体数据,若未命中,则返回200并携带资源实体数据。协商缓存是利用的是Last-Modified,If-Modified-Since和ETag、If-None-Match这两对Header来管理的。

  • 常用的协商缓存响应头包括:
    • Last-Modified:指定资源的最后修改时间。HTTP1.0 提出。浏览器会在请求头加上If-Modified-Since即上次响应的Last-Modified的值,询问服务器在该日期后资源是否有更新,有更新的话就会将新的资源发送回来,但是如果在本地打开缓存文件,就会造成Last-Modified被修改,所以在HTTP 1.1出现了ETag。
    • ETag:指定资源的实体标签(根据内容生成一个哈希),是一个唯一标识符。HTTP1.1 提出,只有资源变化才会被修改(跟最后修改时间没有关系)。If-None-Match的请求头字段会将上次返回的Etag发送给服务器,询问该资源的Etag是否有更新,有变动就会发送新的资源回来。ETag的优先级比Last-Modified更高。
  • 强缓存和协商缓存同时出现,默认强缓存的优先级更高。
  • Last-Modified 和 ETag 的区别:Last-Modified精度只能到秒,但是性能更好。ETag更精细,但是性能不如前者,因为文件变化都要重新计算hash值。
import express from 'express'
import cors from 'cors'
import fs from 'node:fs'
import crypto from 'node:crypto'

const app = express()
app.use(cors())

//获取文件的最后修改时间
const getFileModifyTime = ()=> {
    return fs.statSync('./index.js').mtime.toISOString()
}
app.get('/api3', (req,res)=> {
    // no-cache 使用协商缓存而不是强缓存
    // no-store 不走任何缓存
    res.setHeader('Cache-Control' ,'no-cache')
    const modifyTime = getFileModifyTime();
    // 浏览器根据 Last-Modified 的值自动设置 if-modified-since 的值(与上一次 Last-Modified 的值相同)
    const ifModifiedSince = req.headers['if-modified-since']
    // 如果文件不修改则不用重新缓存
    if (ifModifiedSince === modifyTime) {
        console.log('缓存了')
        res.statusCode = 304
        res.end()
        return
    }
    console.log('没有缓存')
    res.setHeader('Last-Modified', modifyTime)
    res.send('Last-Modified111')
})

app.listen(3000, () => {
    console.log('3000端口已启用')
})
import express from 'express'
import cors from 'cors'
import fs from 'node:fs'
import crypto from 'node:crypto'

const app = express()
app.use(cors())

//获取文件内容哈希
const getFileHash = ()=> {
    return crypto.createHash('sha256').update(fs.readFileSync('index.js')).digest('hex')
}
app.get('/api3', (req,res)=> {
    // no-cache 使用协商缓存而不是强缓存
    // no-store 不走任何缓存
    res.setHeader('Cache-Control' ,'no-cache')
    const fileHash = getFileHash();
    const ifNoneMatch = req.headers['if-none-match']
    // 如果文件不修改则不用重新缓存
    if (ifNoneMatch === fileHash) {
        console.log('缓存了')
        res.statusCode = 304
        res.end()
        return
    }
    console.log('没有缓存')
    res.setHeader('ETag', fileHash)
    res.send('Etag111')
})

app.listen(3000, () => {
    console.log('3000端口已启用')
})

强缓存和协商缓存可以结合使用,以提高缓存效果。通常情况下,可以首先使用强缓存来尽可能减少对服务器的请求,如果资源已经过期或者需要重新验证,则使用协商缓存来验证缓存的有效性。这样可以在保证性能的同时,确保客户端始终能够获取到最新的资源。


http://www.kler.cn/a/461122.html

相关文章:

  • 默认ip无法访问,利用dhcp功能获取ip进行访问的方法
  • mysql 死锁案例及简略分析
  • Node.js 中 http 模块的深度剖析与实战应用
  • 大带宽服务器和普通服务器相比较的优势
  • 国产编辑器EverEdit - 常用资源汇总
  • 基于python大数据的美团外卖的数据分析系统的设计与实现
  • Android 检测设备是否 Root
  • 【数据结构】线性数据结构——栈
  • 本地部署Hello-Algo打造私人算法教练让算法学习告别网络限制
  • 解构大语言模型(LLM)
  • 如何免费解锁 IPhone 网络
  • 如何使用 ChatGPT Prompts 写学术论文?
  • 嵌入式单片机中SPI外设控制与实现
  • 网神SecFox运维安全管理与审计系统 /authService/login接口反序列化漏洞复现 [附POC]
  • Vue.js组件开发-实现多级菜单
  • want php学习笔记
  • 【mysql】linux安装mysql客户端
  • 计算机体系结构期末考试
  • PDF怎么压缩得又小又清晰?5种PDF压缩方法
  • WPF合并C1FlexGrid表格,根据多列的值进行合并
  • JavaWeb开发(二)IDEA创建Java Web项目并部署及目录结构
  • Applied Spatial Statistics(十三)带有空间平滑器的 GAM
  • 批量新建工作表,带个性化模板一步到位-Excel易用宝
  • 12.29~12.31[net][review]need to recite[part 2]
  • 电脑主机后置音频插孔无声?还得Realtek高清晰音频管理器调教
  • 【题解】AT_abc386_d AtCoder Beginner Contest 386 D Diagonal Separation