##瀏覽器的強制緩存和協商緩存
### 強制緩存參數
1. **Cache-Control**:
* **作用**:控制緩存的行為,是 HTTP/1.1 中最重要的緩存相關頭部。
* **常見指令**:
* `max-age=<seconds>`: 指定資源在本地緩存中的最大存活時間。
* `no-cache`: 強制要求緩存服務器在返回資源之前驗證資源的有效性。
* `no-store`: 禁止緩存存儲和提供任何緩存版本。
* **示例**:`Cache-Control: max-age=3600`
2. **Expires**:
* **作用**:指定資源的到期時間,是 HTTP/1.0 中定義的字段。
* **注意**:`Expires`是一個具體的到期日期,而`Cache-Control`中的`max-age`是一個相對時間。
### 協商緩存參數
1. **Last-Modified**:
* **作用**:服務器返回資源的最后修改時間。
* **工作原理**:瀏覽器通過`If-Modified-Since`請求頭將上次獲取資源的`Last-Modified`時間發送給服務器,服務器根據這個時間判斷資源是否有更新。
2. **ETag**:
* **作用**:資源的唯一標識符(Entity Tag)。
* **工作原理**:瀏覽器通過`If-None-Match`請求頭將上次獲取資源的`ETag`發送給服務器,服務器根據這個標識符判斷資源是否有更新。
### 區別總結
* **強制緩存**通過`Cache-Control`和`Expires`字段控制,瀏覽器直接根據這些字段判斷是否從緩存中獲取資源。
* **協商緩存**通過`Last-Modified`和`ETag`字段控制,瀏覽器通過發送上次獲取資源的標識符給服務器,由服務器判斷資源是否需要更新。
### 圖解

### 問答
##### 提問:為什么html文件不需要加header頭控制,瀏覽器訪問時第二次就是304了,而php文件需要自己處理返回304,html文件的304是誰控制的?
當瀏覽器訪問一個 HTML 文件時,服務器通常會自動處理緩存控制和返回 `304 Not Modified` 狀態碼,而 PHP 文件則需要開發者手動管理緩存和條件請求。這種區別源于靜態文件和動態文件的處理方式不同。
### HTML 文件的 304 響應
#### 1. **靜態文件的處理**
HTML 文件通常被視為靜態資源,存儲在服務器的文件系統中。服務器(如 Apache、Nginx 等)在處理靜態文件時,會根據文件的時間戳(例如文件的最后修改時間 `Last-Modified`)和內容的哈希值(如 `ETag`)自動設置響應頭。
當瀏覽器第一次訪問這個 HTML 文件時,服務器會返回文件內容,并在響應頭中包含 `Last-Modified` 和/或 `ETag` 頭。瀏覽器會緩存該文件以及這些頭信息。
#### 2. **瀏覽器的緩存機制**
在隨后的請求中,瀏覽器會在請求頭中附帶 `If-Modified-Since` 或 `If-None-Match` 頭,這些頭信息中包含上次接收到的 `Last-Modified` 或 `ETag` 值。服務器會比較文件的當前時間戳或內容哈希值與請求頭中的值:
- **如果文件沒有變化**(即時間戳或哈希值未變),服務器會返回 `304 Not Modified`,并且不會再次傳輸文件內容。這減少了帶寬消耗。
- **如果文件發生變化**,服務器會返回新的文件內容和狀態碼 `200 OK`。
這一切都是由服務器軟件自動處理的,開發者通常不需要干預。
### PHP 文件的 304 響應
#### 1. **動態文件的處理**
PHP 文件是動態生成的內容,每次請求時 PHP 代碼都會被執行并生成新的頁面內容。因此,PHP 文件的內容可能每次都不同,服務器無法自動確定文件是否“未修改”。
為了支持 `304 Not Modified` 響應,開發者需要在 PHP 代碼中手動實現緩存控制。具體來說:
- **手動設置 `Last-Modified` 或 `ETag`**:開發者需要在 PHP 腳本中設置響應頭,例如 `Last-Modified` 和 `ETag`,以表明內容的生成時間或哈希值。
- **檢查條件請求頭**:在處理請求時,開發者需要手動檢查請求頭中的 `If-Modified-Since` 或 `If-None-Match`,并決定是否返回 `304 Not Modified`。
#### 2. **示例**
以下是一個簡單的 PHP 示例,演示如何處理 `304 Not Modified`:
```php
<?php
// 假設每次生成的內容都有一個固定的時間戳或哈希值
$lastModifiedTime = filemtime(__FILE__); // 這里使用腳本的修改時間作為例子
$etag = md5_file(__FILE__); // 使用文件內容的 MD5 哈希
// 設置響應頭
header("Last-Modified: " . gmdate("D, d M Y H:i:s", $lastModifiedTime) . " GMT");
header("ETag: \"$etag\"");
// 檢查 If-Modified-Since 頭和 If-None-Match 頭
if ((isset($_SERVER['HTTP_IF_MODIFIED_SINCE']) &&
strtotime($_SERVER['HTTP_IF_MODIFIED_SINCE']) == $lastModifiedTime) ||
(isset($_SERVER['HTTP_IF_NONE_MATCH']) &&
trim($_SERVER['HTTP_IF_NONE_MATCH']) == $etag)) {
// 文件未修改,返回 304
header("HTTP/1.1 304 Not Modified");
exit;
}
// 文件已修改或首次請求,生成內容
echo "<html><body><h1>Hello, World!</h1></body></html>";
?>
```
### 總結
- **HTML 文件(靜態資源)**:服務器自動處理緩存控制和 `304 Not Modified` 響應。服務器根據文件的時間戳和內容自動生成 `Last-Modified` 和 `ETag` 頭,并在條件請求時返回 `304`。
- **PHP 文件(動態資源)**:由于每次請求可能返回不同的內容,服務器無法自動處理 `304 Not Modified` 響應。開發者需要手動設置緩存頭并根據條件請求頭決定是否返回 `304`。
這種機制有助于提高性能,減少不必要的帶寬消耗,并確保瀏覽器能夠正確緩存和更新內容。
- 后端
- PHP
- php接收base64格式的圖片
- php 下載文件
- 位,字節,字符的區別
- 求模技巧
- php curl
- php 瀏覽器禁用cookie后需要使用session 就可以用url傳遞session_id
- 有用小方法
- phpDoc
- php 文件鎖來解決高并發
- php小知識
- PHP根據身份證號碼,獲取性別、獲取生日、計算年齡等多個信息
- php 獲取今天,明天、本周、本周末、本月的起始時間戳和結束時間戳的方法
- php 無限級分類
- xdebug設置
- curl
- 獲取現在距離當天結束的還有多少秒
- win10安裝php8版本報錯(Fix PHP Warning: vcruntime140.dll 14.0 is not compatible with this PHP build.)
- 有趣代碼注釋
- php array_diff用法
- parse_str 處理http的query參數
- PHP文件上傳限制
- php操作html
- php trim 函數的使用
- thinkphp5
- 定時任務不能連接數據庫
- 寶塔設置計劃任務
- 控制方法 return $data ,不能直接返回json
- tp5.1命令行
- tp3.2.3 報internal server error
- 悟空crm
- web-msg-sender的使用
- 杉德支付
- laravel
- laravel 遷移文件的使用
- laravel的安裝
- laravel 單元測試
- laravel seeder的使用
- 模型相關
- restful理解
- laravel 的表單驗證
- laravel 隊列的使用
- laravel響應宏應用macro
- laravel 判斷集合是否為空
- laravel 使用ymondesigns/jwt-auth jwt
- laravel 模型工廠
- laravel 自定義助手函數
- laravel 自帶auth的登錄
- 寶塔開啟laravel隊列
- laravel 蘋果內購
- laravel 中的.env.example
- laravel 監聽執行過的sql語句
- laravel-websockets 替代pusher 發送頻道消息
- 記laravel config配置文件目錄中不能使用 url()助手函數
- laravel使用 inspector 進行實時監控
- laravel 項目部署的配置
- laravel 刪除mongodb集合
- laravel 自定義項目命名空間
- laravel 易錯提醒
- laravel 自己組裝分頁
- laravel 設置定時任務
- laravel事件和隊列指定隊列名
- laravel 使用validate檢測名字是否唯一
- laravel + nginx 偽靜態分析
- fastadmin
- cms
- 標簽
- 模板
- dact-admin
- dcat-admin的安裝
- dcat-admin的curd使用
- dact-admin表單使用
- dcat-admin行為表單使用
- dcat-admin使用技巧
- dcat-admin自定義文件上傳
- dcat-admin的js彈窗
- dcat-admin 工具表單傳參
- dcat-admin listbox編輯回顯用法
- weixin
- 微信支付
- 支付類
- 小程序
- 微信提現類
- jwt
- lcobucci/jwt
- Firebase\JWT
- phpstudy
- nginx配置tp5 505 404 錯誤
- tp5重寫 apache
- 織夢模板 使用weight 排序
- phpstudy 添加php8.1版本
- phpstudy ERR_CONNECTION_REFUSED
- phpstudy 設置多個版本php
- 阿里云
- 支付寶支付
- 阿里云短信
- 阿里云OSS上傳圖片報錯
- 阿里云號碼認證(一鍵登錄)
- send login code error: 發送驗證碼失敗:cURL error 28: Connection timed out after 5001 milliseconds
- 極光號碼認證(一鍵登錄)
- git使用
- git
- sentry專欄
- sentry的私有化部署
- sentry設置郵箱
- sentry設置url地址
- sentry中KafkaError OFFSET_OUT_OF_RANGE error
- centos
- tar 壓縮解壓
- centos 8 Errors during downloading metadata for repository 'appstream'
- vim的使用
- ssh秘鑰登錄
- 修改了.bashrc不能立即生效
- 設置軟連接
- 使用echo清空文件內容
- 查看文件大小
- centos8 設置靜態ip
- nginx
- nginx的學習
- nginx配置wss
- supervisor的使用
- shell的使用
- 數據庫
- mysql
- mysql的事務隔離級別
- mysql共享鎖和排它鎖
- mysql的三范式
- mysql 在那些場景下索引會失效
- mysql 的書寫順序
- mysql case用法
- mysql 以逗號分割字符串
- msyql innodb 行鎖解決高并發
- mysql修改字符集
- 鎖
- 樂觀鎖悲觀鎖
- mysql 最左索引原則
- mysql 同表兩列值互換
- mysql升序排列字段為0的在最后
- mysql case when then else end 語法
- mysql 常見錯誤
- mysql json用法
- MongoDB
- mongodb安裝
- redis
- redis 常用通用命令
- string類型的常見命令
- 連接遠程redis刪除指定的值
- markdown
- markdown的使用
- github
- github使用小技巧
- jenkins
- 安裝jenkins
- jenkins設置時區
- docker
- 安裝docker
- docker容器設為自啟動和取消容器自啟動
- docker 安裝mysql
- docker-compose
- docker 安裝php
- docker-compose安裝nginx
- docker-compose安裝php
- docker安裝php+supervisor
- composer使用
- composer
- win10檢查端口占用
- 局域網內同事訪問自己的項目
- 本地測試設置https辦法
- 正則表達式
- 前端代碼和后臺代碼部署在一起的解決方法
- pc微信抓包小程序
- xshell一年后提示需要更新才能打開
- 使用ssh秘鑰登錄服務器
- supervisor
- supervisor的使用
- 瀏覽器的強制緩存和協商緩存
- window11下ssh遠程登錄服務器
- chatgpt
- 注冊chatgpt
- 第三方chatgpt地址
- 前端
- jquery 常用方法
- jquery 省市區三級聯動
- 百度地圖短地址
- npm
- webpack
- vue
- 谷歌安裝vue-devtools的使用
- swiper 一屏顯示頁面
- 騰訊地圖
- jquery點擊圖片放大
- 移動端rem適配
- 彈性布局flex
- CSS
- box-sizing
- 移動端去掉滾動條
- 三角形
- 樹形結構
- require.js的使用
- 微擎人人商城
- 人人商城彈出框
- 常用方法
- 客服消息
- 企業支付到零錢
- 修復權限問題
- 獲取access_token
- 其他管理員沒有應用 調用不了p方法
- 修改公眾號推送消息
- 人人商城
- 人人商城二開常見問題
- 人人商城應用顯示隱藏
- 微擎
- 人人商城小程序解密登錄
- 面試題
- 遍歷目錄中的文件和目錄
- 冒泡排序
- php 在字符串中找到最長對稱字符串
- 地圖相關
- 百度地圖根據ip獲取地址
- 百度,騰訊,高德,地圖點擊跳轉
- 百度地圖根據地址獲取經緯度
- 百度地圖和騰訊地圖經緯度互轉
- 其他
- B站跳過充電環節
- 可愛貓咪回收站制作(附圖)
- 程序員變量命名網站
- 解決谷歌瀏覽器強制跳轉https
- 隨機密碼生成
- 編輯器
- vs code使用
- phpstrom
- phpstrom 常用命令
- phpstrom ctrl+b后想回到之前的位置
- phpstrom 批量操作下劃線轉駝峰
- phpstrom 插件
- phpstrom 使用ctrl+shift+f后搜索不能輸入中文
- phpstrom中項目.env文件會自動消失,不顯示
- vscode插件