成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專(zhuān)欄INFORMATION COLUMN

JavaScript工作原理(八):Service Workers,生命周期和應(yīng)用案例

huaixiaoz / 3125人閱讀

摘要:的生命周期生命周期與您的網(wǎng)頁(yè)是完全分開(kāi)。激活安裝之后,下一步是將其激活。一旦激活,將開(kāi)始控制所有屬于其范圍的頁(yè)面。否則,將執(zhí)行事件。響應(yīng)結(jié)果被添加到緩存中。請(qǐng)求和響應(yīng)必須被克隆,因?yàn)樗鼈兪橇?。新的將啟?dòng)并且事件將被觸發(fā)。


您可能已經(jīng)知道,漸進(jìn)式Web應(yīng)用(PWA)會(huì)越來(lái)越受歡迎,因?yàn)樗鼈冎荚谑筗eb應(yīng)用的用戶(hù)體驗(yàn)更加流暢,創(chuàng)建Native應(yīng)用程序般的體驗(yàn),而不只是運(yùn)行在瀏覽器的應(yīng)用。

構(gòu)建漸進(jìn)式Web應(yīng)用程序的主要原因之一是使應(yīng)用在網(wǎng)絡(luò)和加載方面非常可靠 - 它應(yīng)該可用于不確定或不存在的網(wǎng)絡(luò)條件。

在這篇文章中,我們將深入探討Service Worker:他們?nèi)绾芜\(yùn)作以及應(yīng)該注意的地方。最后,我們還列出了您應(yīng)該利用的Service Workers的一些獨(dú)特優(yōu)勢(shì)。

概述

如果你想了解關(guān)于Service Workers的一切,你應(yīng)該首先閱讀本系列第幾篇文章。

基本上,Service Worker是一種網(wǎng)絡(luò)工作者,更具體地說(shuō),它就像一個(gè)Shared Worker:

Service Worker在它自己的全局腳本上下文中運(yùn)行

它沒(méi)有綁定到特定的網(wǎng)頁(yè)

它無(wú)法訪問(wèn)DOM

Service Worker API令人興奮的主要原因之一是它可以讓你的網(wǎng)絡(luò)應(yīng)用程序支持離線體驗(yàn),從而使開(kāi)發(fā)人員能夠完全控制流程。

Service Worker 的生命周期

Service Worker生命周期與您的網(wǎng)頁(yè)是完全分開(kāi)。它由以下幾個(gè)階段組成:

下載

安裝

激活

下載

這是瀏覽器下載包含Service Worker的.js文件的時(shí)候。

安裝

您的Web應(yīng)用程序想要安裝Service Worker,您必須先注冊(cè)它,您可以在JavaScript代碼中進(jìn)行注冊(cè)。當(dāng)Service Worker被注冊(cè)時(shí),它會(huì)提示瀏覽器在后臺(tái)啟動(dòng)Service Worker安裝步驟。

通過(guò)注冊(cè)Service Worker,你可以告訴瀏覽器你的Service Worker的JavaScript文件在哪里。我們來(lái)看下面的代碼:

if ("serviceWorker" in navigator) {
  window.addEventListener("load", function() {
    navigator.serviceWorker.register("/sw.js").then(function(registration) {
      // 注冊(cè)成功
      console.log("ServiceWorker registration successful");
    }, function(err) {
      // 注冊(cè)失敗
      console.log("ServiceWorker registration failed: ", err);
    });
  });
}

該代碼檢查當(dāng)前環(huán)境中是否支持Service Worker API。如果支持,則注冊(cè)/sw.js Service Worker。

您可以在每次加載頁(yè)面時(shí)調(diào)用register()方法而不用擔(dān)心 - 瀏覽器會(huì)判斷Service Worker是否已經(jīng)注冊(cè),并且會(huì)正確處理。

register()方法的一個(gè)重要細(xì)節(jié)是Service Worker文件的位置。在這種情況下,您可以看到服務(wù)工作者文件位于域的根目錄。這意味著Service Worker的范圍將是整個(gè)來(lái)源。換句話說(shuō),這個(gè)Service Worker將會(huì)收到這個(gè)域的所有東西的fetch事件(我們將在后面討論)。如果我們?cè)?example/sw.js注冊(cè)Service Worker文件,那么服務(wù)工作者將只能看到URL以/example/(即/example/page1/,/example/page2/)開(kāi)頭的頁(yè)面的fetch事件。

在安裝階段,最好加載和緩存一些靜態(tài)資源。資源成功緩存后,Service Worker安裝完成。如果沒(méi)有成功(加載失?。?- Service Worker將重試。一旦安裝成功,您將知道靜態(tài)資源位于緩存中。
在安裝階段,最好加載和緩存一些靜態(tài)資源。資源成功緩存后,Service Worker安裝完成。如果沒(méi)有成功(加載失?。?- Service將重試。一旦安裝成功,您將知道靜態(tài)資產(chǎn)位于緩存中。

如果注冊(cè)需要在加載事件之后發(fā)生,這將回答您的問(wèn)題。這不是必須的,但它是絕對(duì)推薦的。

為什么這樣?讓我們考慮用戶(hù)第一次訪問(wèn)您的網(wǎng)絡(luò)應(yīng)用程序。目前還沒(méi)有Service Worker,瀏覽器無(wú)法事先知道是否會(huì)有安裝的Service Worker。如果安裝了Service Worker,則瀏覽器需要為這個(gè)額外的線程花費(fèi)額外的CPU和內(nèi)存,否則瀏覽器將花費(fèi)在渲染網(wǎng)頁(yè)上。

最重要的是,如果你只是在你的頁(yè)面上安裝一個(gè)Service Worker,你可能會(huì)有延遲加載和渲染的風(fēng)險(xiǎn) - 而不是盡快讓你的用戶(hù)可以使用這個(gè)頁(yè)面。

請(qǐng)注意,這僅在第一次訪問(wèn)頁(yè)面時(shí)才顯得重要。后續(xù)頁(yè)面訪問(wèn)不受Service Worker安裝的影響。一旦在第一次訪問(wèn)頁(yè)面時(shí)激活Service Worker,它可以處理加載/緩存事件,以便隨后訪問(wèn)您的Web應(yīng)用程序。這一切都是有道理的,因?yàn)樗枰獪?zhǔn)備好處理有限的網(wǎng)絡(luò)連接。

激活

安裝Service Worker之后,下一步是將其激活。這一步是管理以前緩存的好機(jī)會(huì)。

一旦激活,Service Worker將開(kāi)始控制所有屬于其范圍的頁(yè)面。一個(gè)有趣的事實(shí)是:首次注冊(cè)Service Worker的頁(yè)面將不會(huì)被控制,直到該頁(yè)面再次被加載。一旦處于Service Worker控制之下,它將處于以下?tīng)顟B(tài)之一:

它將處理從頁(yè)面發(fā)出網(wǎng)絡(luò)請(qǐng)求或消息時(shí)發(fā)生的fetch和message事件

它將被終止以節(jié)省內(nèi)存

以下是生命周期的外觀:

在ervice Worker內(nèi)部處理安裝

在頁(yè)面加速注冊(cè)過(guò)程之后,讓我們看看Service Worker腳本中發(fā)生了什么,它通過(guò)向Service Worker實(shí)例添加事件偵聽(tīng)器來(lái)處理install事件。

這些是install事件處理時(shí)需要采取的步驟:

打開(kāi)緩存

緩存我們的文件

確認(rèn)是否緩存了所有必需的資源

這是一個(gè)簡(jiǎn)單的install在Service Worker內(nèi):

var CACHE_NAME = "my-web-app-cache";
var urlsToCache = [
  "/",
  "/styles/main.css",
  "/scripts/app.js",
  "/scripts/lib.js"
];

self.addEventListener("install", function(event) {
  // event.waitUntil takes a promise to know how
  // long the installation takes, and whether it 
  // succeeded or not.
  event.waitUntil(
    caches.open(CACHE_NAME)
      .then(function(cache) {
        console.log("Opened cache");
        return cache.addAll(urlsToCache);
      })
  );
});

如果所有文件成功被緩存,Service Worker會(huì)被安裝。如果任何文件下載失敗,install步驟會(huì)失敗。所以主要注意你需要緩存的文件。

處理install事件是可選的,你可以避免處理它。在這種情況下,上面三個(gè)步驟你都無(wú)需處理。

在運(yùn)行時(shí)緩存請(qǐng)求

這部分是關(guān)鍵所在。這里您將看到如何攔截請(qǐng)求并返回創(chuàng)建的緩存(并創(chuàng)建新緩存)。

安裝Service Worker并且用戶(hù)導(dǎo)航到另一個(gè)頁(yè)面或刷新他所在的頁(yè)面后,Service Worker將收到fetch事件。下面是一個(gè)演示如何返回緩存資源或執(zhí)行新請(qǐng)求然后緩存結(jié)果的示例:

self.addEventListener("fetch", function(event) {
  event.respondWith(
    // This method looks at the request and
    // finds any cached results from any of the
    // caches that the Service Worker has created.
    caches.match(event.request)
      .then(function(response) {
        // If a cache is hit, we can return thre response.
        if (response) {
          return response;
        }

        // Clone the request. A request is a stream and
        // can only be consumed once. Since we are consuming this
        // once by cache and once by the browser for fetch, we need
        // to clone the request.
        var fetchRequest = event.request.clone();
        
        // A cache hasn"t been hit so we need to perform a fetch,
        // which makes a network request and returns the data if
        // anything can be retrieved from the network.
        return fetch(fetchRequest).then(
          function(response) {
            // Check if we received a valid response
            if(!response || response.status !== 200 || response.type !== "basic") {
              return response;
            }

            // Cloning the response since it"s a stream as well.
            // Because we want the browser to consume the response
            // as well as the cache consuming the response, we need
            // to clone it so we have two streams.
            var responseToCache = response.clone();

            caches.open(CACHE_NAME)
              .then(function(cache) {
                // Add the request to the cache for future queries.
                cache.put(event.request, responseToCache);
              });

            return response;
          }
        );
      })
    );
});

簡(jiǎn)而言之,這就是發(fā)生的過(guò)程:

event.respondWith()將決定我們?nèi)绾雾憫?yīng)fetch事件。我們傳遞了一個(gè)來(lái)自caches.match()的promise,它查看請(qǐng)求并發(fā)現(xiàn)是否有任何已創(chuàng)建的緩存的緩存結(jié)果。

如果存在緩存,則返回結(jié)果。

否則,將執(zhí)行fetch事件。

檢查狀態(tài)是否為200。我們還檢查響應(yīng)類(lèi)型是否basic,這表明它是來(lái)自我們origin的請(qǐng)求。在這種情況下,不會(huì)緩存對(duì)第三方資源的請(qǐng)求。

響應(yīng)結(jié)果被添加到緩存中。

請(qǐng)求和響應(yīng)必須被克隆,因?yàn)樗鼈兪橇?。流的主體只能被使用一次。而且由于我們想要使用它們,瀏覽器也要使用它們,所以必需克隆它們。

更新服務(wù)工作者

當(dāng)用戶(hù)訪問(wèn)您的Web應(yīng)用程序時(shí),瀏覽器會(huì)嘗試重新下載包含Service Worker代碼的.js文件。這發(fā)生在后臺(tái)。

如果與當(dāng)前Service Worker的文件相比,現(xiàn)在下載的Service Worker文件中甚至存在單字節(jié)差異,則瀏覽器將假定有改變并且必須啟動(dòng)新的Service Worker。

新的Service Worker將啟動(dòng)并且install事件將被觸發(fā)。然而,在這一點(diǎn)上,舊的Service Worker仍在控制你的網(wǎng)絡(luò)應(yīng)用程序的頁(yè)面,這意味著新的Service Worker將進(jìn)入waiting狀態(tài)。

一旦您的Web應(yīng)用程序當(dāng)前打開(kāi)的頁(yè)面關(guān)閉,舊的Service Worker將被瀏覽器終止,新安裝的Service Worker將完全控制。這是當(dāng)它的激活事件將被觸發(fā)。

為什么需要這些?為了避免兩個(gè)版本的Web應(yīng)用程序同時(shí)運(yùn)行在不同的選項(xiàng)卡上 - 這在網(wǎng)絡(luò)上實(shí)際上非常常見(jiàn),并且可能會(huì)創(chuàng)建非常糟糕的錯(cuò)誤(例如,在瀏覽器中存儲(chǔ)數(shù)據(jù)時(shí)存在不同模式的情況)。

從緩存中刪除數(shù)據(jù)

activate回調(diào)中最常見(jiàn)的步驟是緩存管理。你現(xiàn)在要這樣做,因?yàn)槿绻阍诎惭b步驟中刪除了所有舊的緩存,舊的Service Workers將突然停止提供緩存中的文件。

下面是一個(gè)示例,您可以從緩存中刪除某些未列入白名單的文件(在這種情況下,其名稱(chēng)中包含page-1或page-2):

self.addEventListener("activate", function(event) {

  var cacheWhitelist = ["page-1", "page-2"];

  event.waitUntil(
    // Retrieving all the keys from the cache.
    caches.keys().then(function(cacheNames) {
      return Promise.all(
        // Looping through all the cached files.
        cacheNames.map(function(cacheName) {
          // If the file in the cache is not in the whitelist
          // it should be deleted.
          if (cacheWhitelist.indexOf(cacheName) === -1) {
            return caches.delete(cacheName);
          }
        })
      );
    })
  );
});
HTTPS要求

在構(gòu)建Web應(yīng)用程序時(shí),您可以通過(guò)本地主機(jī)使用Service Worker,但是一旦將其部署到生產(chǎn)環(huán)境中,則需要準(zhǔn)備好HTTPS(這是您擁有HTTPS的最后一個(gè)原因)。

使用Service Worker,您可以劫持連接并制作響應(yīng)。 通過(guò)不使用HTTPs,您的Web應(yīng)用程序變得容易受到攻擊。

為了讓事情更安全,您需要在通過(guò)HTTPS提供的頁(yè)面上注冊(cè)Service Worker,以便您知道瀏覽器接收的Service Worker在通過(guò)網(wǎng)絡(luò)中未被修改。

瀏覽器支持

服務(wù)人員的瀏覽器支持正在變得越來(lái)越好:

Service Worker應(yīng)用場(chǎng)景

Service Worker提供的一些獨(dú)特功能是:

推送通知 - 允許用戶(hù)選擇從網(wǎng)絡(luò)應(yīng)用程序及時(shí)更新。

后臺(tái)同步 - 允許您推遲操作,直到用戶(hù)具有穩(wěn)定的連接。這樣,您可以確保無(wú)論用戶(hù)想要發(fā)送什么,實(shí)際上都會(huì)發(fā)送。

定期同步(未支持) - 提供管理定期后臺(tái)同步功能的API。

Geofencing(未來(lái)支持) - 您可以定義參數(shù),也稱(chēng)為圍繞感興趣區(qū)域的地理圍欄。當(dāng)設(shè)備跨越地理圍欄時(shí),Web應(yīng)用程序會(huì)收到通知,這可以讓您根據(jù)用戶(hù)的地理位置提供有用的體驗(yàn)。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/107759.html

相關(guān)文章

  • JavaScript 工作原理Service Workers生命周期及其使用場(chǎng)景

    摘要:生命周期的生命周期和網(wǎng)頁(yè)完全不相關(guān)。意即會(huì)作用于整個(gè)源地址上。激活安裝完之后下一步即激活。同時(shí)檢查響應(yīng)類(lèi)型是否為,即檢查請(qǐng)求是否同域。創(chuàng)建新的的過(guò)程將會(huì)啟動(dòng),然后觸發(fā)事件。可以利用劫持網(wǎng)絡(luò)連接和偽造響應(yīng)數(shù)據(jù)。 原文請(qǐng)查閱這里,略有刪減,本文采用知識(shí)共享署名 4.0 國(guó)際許可協(xié)議共享,BY Troland。 本系列持續(xù)更新中,Github 地址請(qǐng)查閱這里。 這是 JavaScript 工...

    oysun 評(píng)論0 收藏0
  • JavaScript 是如何工作的:Service Worker 的生命周期及使用場(chǎng)景

    摘要:的生命周期的生命周期與頁(yè)面完全分離。換句話說(shuō),這個(gè)將為這個(gè)域中的所有內(nèi)容接收事件。這不是必要的,但絕對(duì)是推薦的。新的將啟動(dòng)并且安裝事件將被移除。使用,可以很容易被劫持連接并偽造響應(yīng)。后臺(tái)同步允許延遲操作,直到用戶(hù)具有穩(wěn)定的連接。 這是專(zhuān)門(mén)探索 JavaScript 及其所構(gòu)建的組件的系列文章的第8篇。 想閱讀更多優(yōu)質(zhì)文章請(qǐng)猛戳GitHub博客,一年百來(lái)篇優(yōu)質(zhì)文章等著你! 如果你錯(cuò)過(guò)了前...

    zzzmh 評(píng)論0 收藏0
  • 構(gòu)建離線web應(yīng)用(一)

    摘要:我喜歡移動(dòng),而且也是那些堅(jiān)持使用技術(shù)構(gòu)建移動(dòng)應(yīng)用程序的人之一。我們準(zhǔn)備做這樣的一個(gè)漸進(jìn)式應(yīng)用是典型的旨在提高用戶(hù)離線體驗(yàn)的應(yīng)用。當(dāng)我們開(kāi)始構(gòu)建應(yīng)用時(shí),你就能理解上面的場(chǎng)景了。的作用范圍是針對(duì)相對(duì)路徑的。最佳的做法是在應(yīng)用的入口。 我喜歡移動(dòng)app,而且也是那些堅(jiān)持使用Web技術(shù)構(gòu)建移動(dòng)應(yīng)用程序的人之一。 經(jīng)過(guò)技術(shù)的不斷迭代(可能還有一些其它的東西),移動(dòng)體驗(yàn)設(shè)計(jì)愈來(lái)愈平易近人,給予用戶(hù)...

    Sanchi 評(píng)論0 收藏0
  • 聊聊webWorker

    摘要:不過(guò),這并不意味著語(yǔ)言本身就支持了多線程,對(duì)于語(yǔ)言本身它仍是運(yùn)行在單線程上的,只是瀏覽器宿主環(huán)境提供的一個(gè)能力。主線程與子線程之間也可以交換二進(jìn)制數(shù)據(jù),比如等對(duì)象,也可以在線程之間發(fā)送。 先看幾個(gè)例子 本例子是通過(guò)通過(guò)紅點(diǎn)展示地球上的地震帶,數(shù)據(jù)來(lái)自于地質(zhì)探測(cè)局通過(guò)console.log看到數(shù)據(jù)運(yùn)算所耗的時(shí)間不使用 webworker No web workers - all on ...

    luffyZh 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

閱讀需要支付1元查看
<