為什麼重覆的GET請求變慢了?

来源:https://www.cnblogs.com/fundebug/archive/2019/07/17/chrome-stall-multiple-same-request.html
-Advertisement-
Play Games

最近在研究慢請求監控的問題,寫了一個簡單的測試代碼:在網頁端( )通過 函數向服務端獲取數據,然後列印請求耗時。 在服務端通過 延時 才返回數據(服務端使用 "ExpressJS" )。 不出所料, 數據都略微大於 1500。 而後,我突發奇想,假設我同時發送多個請求會怎麼樣呢?於是有瞭如下代碼: ...


最近在研究慢請求監控的問題,寫了一個簡單的測試代碼:在網頁端(index.html)通過fetch函數向服務端獲取數據,然後列印請求耗時。

function requestData() {
    let start = new Date();
    fetch("http://localhost:3000/company/basic")
        .then(res => {
            return res.json();
        })
        .then(res => {
            let span = new Date() - start;
            console.log("span:", span);
        });
}
requestData();

在服務端通過setTimeout延時1500s才返回數據(服務端使用ExpressJS)。

app.get("/company/basic", (req, res) => {
    setTimeout(function() {
        res.send({ hello: "Hello Fundebug!" });
    }, 1500);
});

不出所料,span數據都略微大於 1500。

而後,我突發奇想,假設我同時發送多個請求會怎麼樣呢?於是有瞭如下代碼:

[1, 2, 3].forEach(function() {
    requestData();
});

結果好像也沒問題,在 Chrome 瀏覽器下麵是這個效果:

接入 Fundebug 慢請求監控測試

於是愉快地接入 Fundebug 監控:

<script
    src="https://js.fundebug.cn/fundebug.1.9.0.min.js"
    apikey="API-KEY"
></script>

並設置如果請求時長超過 2 秒就上報:

if ("fundebug" in window) {
    fundebug.httpTimeout = 2000;
}

本以為刷新頁面,應該不會收到報錯。

結果,萬萬沒想到的是,Fundebug 收到 2 個慢請求報錯。

這不科學啊!

點開錯誤詳情,可以看到具體的報錯信息。一個請求耗時 3018 毫秒,一個請求耗時 4525 毫秒。

也就是說,第一個請求沒問題,假設是 1500 毫秒。我們把三個請求的時間放一起看看有何規律:1500,3018,4524。他們近似成等差數列,相差 1500 毫秒。於是,我懷疑三個請求是一個一個阻塞式的,而不是併發的。

測試併發請求不同 API 的情況

為了驗證這一點,我將測試改為請求三個不同的 API 介面。

服務端代碼:

app.get("/company/basic", resp);
app.get("/company/basic1", resp);
app.get("/company/basic2", resp);

function resp(req, res) {
    setTimeout(function() {
        res.send({ hello: "Hello Fundebug!" });
    }, 1500);
}

網頁端代碼(requestData函數傳入請求的 URL):

[
    "http://localhost:3000/company/basic",
    "http://localhost:3000/company/basic1",
    "http://localhost:3000/company/basic2"
].forEach(function(item) {
    requestData(item);
});

為了獲取請求數據,將httpTimeout改為 1500。

if ("fundebug" in window) {
    fundebug.httpTimeout = 1500;
}

Fundebug 捕獲三個請求的時間,分別為 1526,1525,1529。

至此大體驗證了剛剛的假設:對同一個 API 介面的併發請求會被阻塞,對不同的 API 介面併發請求正常執行。

那麼為什麼會被阻塞呢?意圖何在?接下來慢慢給各位介紹。

背後的原因

StackOverflow上找到了答案:

Yes, this behavior is due to Chrome locking the cache and waiting to see the result of one request before requesting the same resource again. The answer is to find a way to make the requests unique.

也就是說,Chrome 特意做了這樣的設計。對於連續的相同請求,Chrome 會阻塞後面的請求,直到前面的完成。通過判斷前面的請求返回的 Header 裡面的緩存設置來決定下一步的行動。

我們可以做個實驗來驗證一下。

緩存實驗

  • 服務端設置緩存 2 秒

    在服務端的介面返回代碼中配置緩存時間

    res.setHeader("Cache-Control", "public, max-age=2");

  • 服務端設置不緩存

    res.setHeader(
        "Cache-Control",
        "private, no-cache, no-store, must-revalidate"
    );

  • Chrome 開發者面板設置Disable Cache

最後的疑問

為什麼打開和不打開谷歌開發者控制台,行為會不一樣了?

其實是有原因的,而且這個干擾項一度成功阻止了我發現問題的本質。當我們在開發前端項目的時候,代碼的改動希望能夠實時地反應到網頁上,而不是受到瀏覽器緩存的影響,但是我們發現往往刷新頁面的時候沒有真的去服務端獲取數據,還是老的信息。於是,我們會去配置一個選項,將Disable Cache設置為true。也就是說,在開發環境下,緩存是被禁用了的,也就不存在等待第一個請求返回然後判斷其 Header 裡面Cache-Control設置的問題。這也是為什麼打開谷歌開發者控制台,請求沒有等待,立即執行了。

關於Fundebug

Fundebug專註於JavaScript、微信小程式、微信小游戲、支付寶小程式、React Native、Node.js和Java線上應用實時BUG監控。 自從2016年雙十一正式上線,Fundebug累計處理了10億+錯誤事件,付費客戶有陽光保險、核桃編程、荔枝FM、掌門1對1、微脈、青團社等眾多品牌企業。歡迎大家免費試用

img

版權聲明

轉載時請註明作者 Fundebug以及本文地址:
https://blog.fundebug.com/2019/07/17/chrome-stall-multiple-same-request/


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 摘要: mpvue中頁面之間傳值(註意:是頁面之間,不是組件之間) 場景:A頁面跳轉B頁面,在B頁面選擇商品,將商品名帶回A頁面並顯示 使用api: getCurrentPages step1: A頁面js: 先定義一個全局的對象that,然後在mouted中把this賦給that step2: B ...
  • 摘要: 小程式索引選擇器,點擊跳轉相應條目,索引可滑動,滑動也可跳轉 場景:城市選擇列表, 汽車品牌選擇列表 所用組件: scroll-view(小程式原生) https://developers.weixin.qq.com/miniprogram/dev/component/scroll-view ...
  • 微信小程式開發實戰教程 一、微信小程式 它是一種混合開發的方式。 是安裝在微信中的程式(一個程式最多2M空間)。 1.1 註冊 1 2 點擊立即註冊:進入下方頁面 3 4 點擊小程式進入表單填寫頁面 5 6 填寫完畢之後提交,會讓你去郵箱中激活。激活之後就可以進入小程式開發了。 1.2 安裝開發工具 ...
  • 在JavaScript中,函數其實就是對象。使函數不同於其他對象的決定性特點是函數存在一個被稱為[[Call]]的內部屬性。內部屬性無法通過代碼訪問而是定義了代碼執行時的行為。ECMAScript為JavaScript的對象定義了多種內部屬性,這些內部屬性都用雙重中括弧來標註。 ​[[Call]]屬 ...
  • wxss文件樣式 .item-image{ width: 80px; height: 80px; margin-right: 2px; } .item-image{ width: 80px; height: 80px; margin-right: 2px; } wxml 佈局 ...
  • 設置錨點的兩種方式 1.設置一個錨點鏈接<a href="#wang">去找汪星人</a> 在頁面中需要的位置設置錨點<a name="wang">汪星人基地</a> 2.設置一個錨點鏈接<a href="#miao">去找喵星人</a> 在頁面中需要的位置設置錨點<h3 id="miao">喵星人 ...
  • 主要介紹了一些數據保存在本地中的一些處理方法,包括cookie和webSrorage的保存優缺點介紹,以及簡單介紹cookie中的坑 ...
  • 我們定義一個組件的時候,可以在組件的某個節點內預留一個位置,當父組件調用該組件的時候可以指定該位置具體的內容,這就是插槽的用法,子組件模板可以通過slot標簽(插槽)規定對應的內容放置在哪裡,比如: 渲染結果為: 對應的html節點如下: 引用AppLayout這個組件時,我們指定了header和f ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...