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

資訊專欄INFORMATION COLUMN

【PWA學(xué)習(xí)與實踐】(3) 讓你的WebApp離線可用

since1986 / 3583人閱讀

摘要:學(xué)習(xí)與實踐系列文章已整理至學(xué)習(xí)手冊,文字內(nèi)容已同步至。本文是學(xué)習(xí)與實踐系列的第三篇文章。引言其中一個令人著迷的能力就是離線可用。但是,如果你注意到文章開頭的圖片就會發(fā)現(xiàn),離線時我們不僅可以訪問,還可以使用搜索功能。

《PWA學(xué)習(xí)與實踐》系列文章已整理至gitbook - PWA學(xué)習(xí)手冊,文字內(nèi)容已同步至learning-pwa-ebook。轉(zhuǎn)載請注明作者與出處。

本文是《PWA學(xué)習(xí)與實踐》系列的第三篇文章。文中的代碼都可以在learning-pwa的sw-cache分支上找到(git clone后注意切換到sw-cache分支)。

PWA作為時下最火熱的技術(shù)概念之一,對提升Web應(yīng)用的安全、性能和體驗有著很大的意義,非常值得我們?nèi)チ私馀c學(xué)習(xí)。對PWA感興趣的朋友歡迎關(guān)注《PWA學(xué)習(xí)與實踐》系列文章。

1. 引言

PWA其中一個令人著迷的能力就是離線(offline)可用。

離線只是它的一種功能表現(xiàn)而已,具體說來,它可以:

讓我們的Web App在無網(wǎng)(offline)情況下可以訪問,甚至使用部分功能,而不是展示“無網(wǎng)絡(luò)連接”的錯誤頁;

讓我們在弱網(wǎng)的情況下,能使用緩存快速訪問我們的應(yīng)用,提升體驗;

在正常的網(wǎng)絡(luò)情況下,也可以通過各種自發(fā)控制的緩存方式來節(jié)省部分請求帶寬;

……

而這一切,其實都要歸功于PWA背后的英雄 —— Service Worker

那么,Service Worker是什么呢?你可以把Service Worker簡單理解為一個獨立于前端頁面,在后臺運行的進程。因此,它不會阻塞瀏覽器腳本的運行,同時也無法直接訪問瀏覽器相關(guān)的API(例如:DOM、localStorage等)。此外,即使在離開你的Web App,甚至是關(guān)閉瀏覽器后,它仍然可以運行。它就像是一個在Web應(yīng)用背后默默工作的勤勞小蜜蜂,處理著緩存、推送、通知與同步等工作。所以,要學(xué)習(xí)PWA,繞不開的就是Service Worker。

在接下來的幾篇文章里,我會從如何使用Service Worker來實現(xiàn)資源的緩存、消息的推送、消息的通知以及后臺同步這幾個角度,來介紹相關(guān)原理與技術(shù)實現(xiàn)。這些部分會是PWA技術(shù)的重點。需要特別注意的是,由于Service Worker所具有的強大能力,因此規(guī)范規(guī)定,Service Worker只能運行在HTTPS域下。然而我們開發(fā)時候沒有HTTPS怎么辦?別著急,還有一個貼心的地方——為方便本地開發(fā),Service Worker也可以運行在localhost(127.0.0.1)域下。

好了,簡單了解了Service Worker與它能實現(xiàn)的功能后,我們還是要回到這一篇的主題,也就是Service Worker的第一部分——如何利用Service Worker來實現(xiàn)前端資源的緩存,從而提升產(chǎn)品的訪問速度,做到離線可用。

2. Service Worker是如何實現(xiàn)離線可用的?

這一小節(jié)會告訴大家,Service Worker是如何讓我們在離線的情況下也能訪問Web App的。當然,離線訪問只是其中一種表現(xiàn)。

首先,我們想一下,當訪問一個web網(wǎng)站時,我們實際上做了什么呢?總體上來說,我們通過與與服務(wù)器建立連接,獲取資源,然后獲取到的部分資源還會去請求新的資源(例如html中使用的css、js等)。所以,粗粒度來說,我們訪問一個網(wǎng)站,就是在獲取/訪問這些資源。

可想而知,當處于離線或弱網(wǎng)環(huán)境時,我們無法有效訪問這些資源,這就是制約我們的關(guān)鍵因素。因此,一個最直觀的思路就是:如果我們把這些資源緩存起來,在某些情況下,將網(wǎng)絡(luò)請求變?yōu)楸镜卦L問,這樣是否能解決這一問題?是的。但這就需要我們有一個本地的cache,可以靈活地將各類資源進行本地存取。

有了本地的cache還不夠,我們還需要能夠有效地使用緩存、更新緩存與清除緩存,進一步應(yīng)用各種個性化的緩存策略。而這就需要我們有個能夠控制緩存的“worker”——這也就是Service Worker的部分工作之一。順便多說一句,可能有人還記得 ApplicationCache 這個API。當初它的設(shè)計同樣也是為了實現(xiàn)Web資源的緩存,然而就是因為不夠靈活等各種缺陷,如今已被Service Worker與cache API所取代了。

Service Worker有一個非常重要的特性:你可以在Service Worker中監(jiān)聽所有客戶端(Web)發(fā)出的請求,然后通過Service Worker來代理,向后端服務(wù)發(fā)起請求。通過監(jiān)聽用戶請求信息,Service Worker可以決定是否使用緩存來作為Web請求的返回。

下圖展示普通Web App與添加了Service Worker的Web App在網(wǎng)絡(luò)請求上的差異:

這里需要強調(diào)一下,雖然圖中好像將瀏覽器、SW(Service Worker)與后端服務(wù)三者并列放置了,但實際上瀏覽器(你的Web應(yīng)用)和SW都是運行在你的本機上的,所以這個場景下的SW類似一個“客戶端代理”。

了解了基本概念之后,就可以具體來看下,我們?nèi)绾螒?yīng)用這個技術(shù)來實現(xiàn)一個離線可用的Web應(yīng)用。

3. 如何使用Service Worker實現(xiàn)離線可用的“秒開”應(yīng)用

還記得我們之前的那個圖書搜索的demo Web App么?不了解的朋友可以看下本系列的第一篇文章,當然你可以忽略細節(jié),繼續(xù)往下了解技術(shù)原理。

沒錯,這次我仍然會基于它進行改造。在上一篇添加了manifest后,它已經(jīng)擁有了自己的桌面圖標,并有一個很像Native App的外殼;而今天,我會讓它變得更酷。

如果想要跟著文章內(nèi)容一起實踐,可以在這里下載到所需的全部代碼。
記得切換到manifest分支,因為本篇內(nèi)容,是基于上一篇的最終代碼進行相應(yīng)的開發(fā)與升級。畢竟我們的最終目標是將這個普通的“圖書搜索”demo升級為PWA。
3.1. 注冊Service Worker

注意,我們的應(yīng)用始終應(yīng)該是漸進可用的,在不支持Service Worker的環(huán)境下,也需要保證其可用性。要實現(xiàn)這點,可以通過特性檢測,在index.js中來注冊我們的Service Worker(sw.js):

// index.js
// 注冊service worker,service worker腳本文件為sw.js
if ("serviceWorker" in navigator) {
    navigator.serviceWorker.register("./sw.js").then(function () {
        console.log("Service Worker 注冊成功");
    });
}

這里我們將sw.js文件注冊為一個Service Worker,注意文件的路徑不要寫錯了。

值得一提的是,Service Worker的各類操作都被設(shè)計為異步,用以避免一些長時間的阻塞操作。這些API都是以Promise的形式來調(diào)用的。所以你會在接下來的各段代碼中不斷看到Promise的使用。如果你完全不了解Promise,可以先在這里了解基本的Promise概念:Promise(MDN)和JavaScript Promise:簡介。

3.2. Service Worker的生命周期

當我們注冊了Service Worker后,它會經(jīng)歷生命周期的各個階段,同時會觸發(fā)相應(yīng)的事件。整個生命周期包括了:installing --> installed --> activating --> activated --> redundant。當Service Worker安裝(installed)完畢后,會觸發(fā)install事件;而激活(activated)后,則會觸發(fā)activate事件。

下面的例子監(jiān)聽了install事件:

// 監(jiān)聽install事件
self.addEventListener("install", function (e) {
    console.log("Service Worker 狀態(tài): install");
});

self是Service Worker中一個特殊的全局變量,類似于我們最常見的window對象。self引用了當前這個Service Worker。

3.3. 緩存靜態(tài)資源

通過上一節(jié),我們已經(jīng)學(xué)會了如何添加事件監(jiān)聽,來在合適的時機觸發(fā)Service Worker的相應(yīng)操作?,F(xiàn)在,要使我們的Web App離線可用,就需要將所需資源緩存下來。我們需要一個資源列表,當Service Worker被激活時,會將該列表內(nèi)的資源緩存進cache。

// sw.js
var cacheName = "bs-0-2-0";
var cacheFiles = [
    "/",
    "./index.html",
    "./index.js",
    "./style.css",
    "./img/book.png",
    "./img/loading.svg"
];

// 監(jiān)聽install事件,安裝完成后,進行文件緩存
self.addEventListener("install", function (e) {
    console.log("Service Worker 狀態(tài): install");
    var cacheOpenPromise = caches.open(cacheName).then(function (cache) {
        return cache.addAll(cacheFiles);
    });
    e.waitUntil(cacheOpenPromise);
});

可以看到,首先在cacheFiles中我們列出了所有的靜態(tài)資源依賴。注意其中的"/",由于根路徑也可以訪問我們的應(yīng)用,因此不要忘了將其也緩存下來。當Service Worker install時,我們就會通過caches.open()cache.addAll()方法將資源緩存起來。這里我們給緩存起了一個cacheName,這個值會成為這些緩存的key。

上面這段代碼中,caches是一個全局變量,通過它我們可以操作Cache相關(guān)接口。

Cache 接口提供緩存的 Request / Response 對象對的存儲機制。Cache 接口像 workers 一樣, 是暴露在 window 作用域下的。盡管它被定義在 service worker 的標準中,  但是它不必一定要配合 service worker 使用?!狹DN
3.4 使用緩存的靜態(tài)資源

到目前為止,我們僅僅是注冊了一個Service Worker,并在其install時緩存了一些靜態(tài)資源。然而,如果這時運行這個demo你會發(fā)現(xiàn)——“圖書搜索”這個Web App依然無法離線使用。

為什么呢?因為我們僅僅緩存了這些資源,然而瀏覽器并不知道需要如何使用它們;換言之,瀏覽器仍然會通過向服務(wù)器發(fā)送請求來等待并使用這些資源。那怎么辦?

聰明的你應(yīng)該想起來了,我們在文章前半部分介紹Service Worker時提到了“客戶端代理”——用Service Worker來幫我們決定如何使用緩存。

下圖是一個簡單的策略:

瀏覽器發(fā)起請求,請求各類靜態(tài)資源(html/js/css/img);

Service Worker攔截瀏覽器請求,并查詢當前cache;

若存在cache則直接返回,結(jié)束;

若不存在cache,則通過fetch方法向服務(wù)端發(fā)起請求,并返回請求結(jié)果給瀏覽器

// sw.js
self.addEventListener("fetch", function (e) {
    // 如果有cache則直接返回,否則通過fetch請求
    e.respondWith(
        caches.match(e.request).then(function (cache) {
            return cache || fetch(e.request);
        }).catch(function (err) {
            console.log(err);
            return fetch(e.request);
        })
    );
});

fetch事件會監(jiān)聽所有瀏覽器的請求。e.respondWith()方法接受Promise作為參數(shù),通過它讓Service Worker向瀏覽器返回數(shù)據(jù)。caches.match(e.request)則可以查看當前的請求是否有一份本地緩存:如果有緩存,則直接向瀏覽器返回cache;否則Service Worker會向后端服務(wù)發(fā)起一個fetch(e.request)的請求,并將請求結(jié)果返回給瀏覽器。

到目前為止,運行我們的demo:當?shù)谝宦?lián)網(wǎng)打開“圖書搜索”Web App后,所依賴的靜態(tài)資源就會被緩存在本地;以后再訪問時,就會使用這些緩存而不發(fā)起網(wǎng)絡(luò)請求。因此,即使在無網(wǎng)情況下,我們似乎依舊能“訪問”該應(yīng)用。

3.5. 更新靜態(tài)緩存資源

然而,如果你細心的話,會發(fā)現(xiàn)一個小問題:當我們將資源緩存后,除非注銷(unregister)sw.js、手動清除緩存,否則新的靜態(tài)資源將無法緩存。

解決這個問題的一個簡單方法就是修改cacheName。由于瀏覽器判斷sw.js是否更新是通過字節(jié)方式,因此修改cacheName會重新觸發(fā)install并緩存資源。此外,在activate事件中,我們需要檢查cacheName是否變化,如果變化則表示有了新的緩存資源,原有緩存需要刪除。

// sw.js
// 監(jiān)聽activate事件,激活后通過cache的key來判斷是否更新cache中的靜態(tài)資源
self.addEventListener("activate", function (e) {
    console.log("Service Worker 狀態(tài): activate");
    var cachePromise = caches.keys().then(function (keys) {
        return Promise.all(keys.map(function (key) {
            if (key !== cacheName) {
                return caches.delete(key);
            }
        }));
    })
    e.waitUntil(cachePromise);
    return self.clients.claim();
});
3.6. 緩存API數(shù)據(jù)的“離線搜索”

到這里,我們的應(yīng)用基本已經(jīng)完成了離線訪問的改造。但是,如果你注意到文章開頭的圖片就會發(fā)現(xiàn),離線時我們不僅可以訪問,還可以使用搜索功能。

這是怎么回事呢?其實這背后的秘密就在于,這個Web App也會把XHR請求的數(shù)據(jù)緩存一份。而再次請求時,我們會優(yōu)先使用本地緩存(如果有緩存的話);然后向服務(wù)端請求數(shù)據(jù),服務(wù)端返回數(shù)據(jù)后,基于該數(shù)據(jù)替換展示。大致過程如下:

首先我們改造一下前一節(jié)的代碼在sw.js的fetch事件里進行API數(shù)據(jù)的緩存

// sw.js
var apiCacheName = "api-0-1-1";
self.addEventListener("fetch", function (e) {
    // 需要緩存的xhr請求
    var cacheRequestUrls = [
        "/book?"
    ];
    console.log("現(xiàn)在正在請求:" + e.request.url);

    // 判斷當前請求是否需要緩存
    var needCache = cacheRequestUrls.some(function (url) {
        return e.request.url.indexOf(url) > -1;
    });

    /**** 這里是對XHR數(shù)據(jù)緩存的相關(guān)操作 ****/
    if (needCache) {
        // 需要緩存
        // 使用fetch請求數(shù)據(jù),并將請求結(jié)果clone一份緩存到cache
        // 此部分緩存后在browser中使用全局變量caches獲取
        caches.open(apiCacheName).then(function (cache) {
            return fetch(e.request).then(function (response) {
                cache.put(e.request.url, response.clone());
                return response;
            });
        });
    }
    /* ******************************* */

    else {
        // 非api請求,直接查詢cache
        // 如果有cache則直接返回,否則通過fetch請求
        e.respondWith(
            caches.match(e.request).then(function (cache) {
                return cache || fetch(e.request);
            }).catch(function (err) {
                console.log(err);
                return fetch(e.request);
            })
        );
    }
});

這里,我們也為API緩存的數(shù)據(jù)創(chuàng)建一個專門的緩存位置,key值為變量apiCacheName。在fetch事件中,我們首先通過對比當前請求與cacheRequestUrls來判斷是否是需要緩存的XHR請求數(shù)據(jù),如果是的話,就會使用fetch方法向后端發(fā)起請求。

fetch.then中我們以請求的URL為key,向cache中更新了一份當前請求所返回數(shù)據(jù)的緩存:cache.put(e.request.url, response.clone())。這里使用.clone()方法拷貝一份響應(yīng)數(shù)據(jù),這樣我們就可以對響應(yīng)緩存進行各類操作而不用擔心原響應(yīng)信息被修改了。

3.7. 應(yīng)用離線XHR數(shù)據(jù),完成“離線搜索”,提升響應(yīng)速度

如果你跟著做到了這一步,那么恭喜你,距離我們酷酷的離線應(yīng)用還差最后一步了!

目前為止,我們對Service Worker(sw.js)的改造已經(jīng)完畢了。最后只剩下如何在XHR請求時有策略的使用緩存了,這一部分的改造全部集中于index.js,也就是我們的前端腳本。

還是回到上一節(jié)的這張圖:

和普通情況不同,這里我們的前端瀏覽器會首先去嘗試獲取緩存數(shù)據(jù)并使用其來渲染界面;同時,瀏覽器也會發(fā)起一個XHR請求,Service Worker通過將請求返回的數(shù)據(jù)更新到存儲中的同時向前端Web應(yīng)用返回數(shù)據(jù)(這一步分就是上一節(jié)提到的緩存策略);最終,如果判斷返回的數(shù)據(jù)與最開始取到的cache不一致,則重新渲染界面,否則忽略。

為了是代碼更清晰,我們將原本的XHR請求部分多帶帶剝離出來,作為一個方法getApiDataRemote()以供調(diào)用,同時將其改造為了Promise。為了節(jié)省篇幅,我部分的代碼比較簡單,就不多帶帶貼出了。

這一節(jié)最重要的部分其實是讀取緩存。我們知道,在Service Worker中是可以通過caches變量來訪問到緩存對象的。令人高興的是,在我們的前端應(yīng)用中,也仍然可以通過caches來訪問緩存。當然,為了保證漸進可用,我們需要先進行判斷"caches" in window。為了代碼的統(tǒng)一,我將獲取該請求的緩存數(shù)據(jù)也封裝成了一個Promise方法:

function getApiDataFromCache(url) {
    if ("caches" in window) {
        return caches.match(url).then(function (cache) {
            if (!cache) {
                return;
            }
            return cache.json();
        });
    }
    else {
        return Promise.resolve();
    }
}

而原本我們在queryBook()方法中,我們會請求后端數(shù)據(jù),然后渲染頁面;而現(xiàn)在,我們加上基于緩存的渲染:

function queryBook() {
    // ……
    // 遠程請求
    var remotePromise = getApiDataRemote(url);
    var cacheData;
    // 首先使用緩存數(shù)據(jù)渲染
    getApiDataFromCache(url).then(function (data) {
        if (data) {
            loading(false);
            input.blur();            
            fillList(data.books);
            document.querySelector("#js-thanks").style = "display: block";
        }
        cacheData = data || {};
        return remotePromise;
    }).then(function (data) {
        if (JSON.stringify(data) !== JSON.stringify(cacheData)) {
            loading(false);                
            input.blur();
            fillList(data.books);
            document.querySelector("#js-thanks").style = "display: block";
        }
    });
    // ……
}

如果getApiDataFromCache(url).then返回緩存數(shù)據(jù),則使用它先進行渲染。而當remotePromise的數(shù)據(jù)返回時,與cacheData進行比對,只有在數(shù)據(jù)不一致時需要重新渲染頁面(注意這里為了簡便,粗略地使用了JSON.stringify()方法進行對象間的比較)。這么做有兩個優(yōu)勢:

離線可用。如果我們之前訪問過某些URL,那么即使在離線的情況下,重復(fù)相應(yīng)的操作依然可以正常展示頁面;

優(yōu)化體驗,提高訪問速度。讀取本地cache耗時相比于網(wǎng)絡(luò)請求是非常低的,因此就會給我們的用戶一種“秒開”、“秒響應(yīng)”的感覺。

4. 使用Lighthouse測試我們的應(yīng)用

至此,我們完成了PWA的兩大基本功能:Web App Manifest和Service Worker的離線緩存。這兩大功能可以很好地提升用戶體驗與應(yīng)用性能。我們用Chrome中的Lighthouse來檢測一下目前的應(yīng)用:

可以看到,在PWA評分上,我們的這個Web App已經(jīng)非常不錯了。其中唯一個扣分項是在HTTPS協(xié)議上:由于是本地調(diào)試,所以使用了http://127.0.0.1:8085,在生產(chǎn)肯定會替換為HTTPS。

5. 這太酷了,但是兼容性呢?

隨著今年(2018年)年初,Apple在iOS 11.3中開始支持Service Worker,加上Apple一直以來較為良好的系統(tǒng)升級率,整個PWA在兼容性問題上有了重大的突破。

雖然Service Worker中的一些其他功能(例如推送、后臺同步)Apple并未表態(tài),但是Web App Manifest和Service Worker的離線緩存是iOS 11.3所支持的。這兩大核心功能不僅效果拔群,而且目前看來具有還不錯的兼容性,非常適合投入生產(chǎn)。

更何況,作為漸進式網(wǎng)頁應(yīng)用,其最重要的一個特點就是在兼容性支持時自動升級功能與體驗;而在不支持時,會靜默回退部分新功能。在保證我們的正常服務(wù)情況下,盡可能利用瀏覽器特性,提供更優(yōu)質(zhì)的服務(wù)。

6. 寫在最后

本文中所有的代碼示例均可以在learn-pwa/sw-cache上找到。注意在git clone之后,切換到sw-cache分支,本文所有的代碼均存在于該分支上。切換其他分值可以看到不同的版本:

basic分支:基礎(chǔ)項目demo,一個普通的圖書搜索應(yīng)用(網(wǎng)站);

manifest分支:基于basic分支,添加manifest等功能,具體可以看上一篇文章了解;

sw-cache分支:基于manifest分支,添加緩存與離線功能;

master分支:應(yīng)用的最新代碼。

如果你喜歡或想要了解更多的PWA相關(guān)知識,歡迎關(guān)注我,關(guān)注《PWA學(xué)習(xí)與實踐》系列文章。我會總結(jié)整理自己學(xué)習(xí)PWA過程的遇到的疑問與技術(shù)點,并通過實際代碼和大家一起實踐。

最后聲明一下,文中的代碼作為demo,主要是用于了解與學(xué)習(xí)PWA技術(shù)原理,可能會存在一些不完善的地方,因此,不建議直接使用到生產(chǎn)環(huán)境。
《PWA技術(shù)學(xué)習(xí)與實踐》系列

第一篇:開始你的PWA學(xué)習(xí)之旅

第二篇:10分鐘學(xué)會使用Manifest,讓你的WebApp更“Native”

第三篇:從今天起,讓你的WebApp離線可用(本文)

第四篇:TroubleShooting: 解決FireBase login驗證失敗問題

第五篇:與你的用戶保持聯(lián)系: Web Push功能

第六篇:How to Debug? 在chrome中調(diào)試你的PWA

第七篇:增強交互:使用Notification API來進行提醒

第八篇:使用Service Worker進行后臺數(shù)據(jù)同步

第九篇:PWA實踐中的問題與解決方案

第十篇:Resource Hint - 提升頁面加載性能與體驗

第十一篇:從PWA離線工具集workbox中學(xué)習(xí)各類離線策略(寫作中…)

參考資料

Using Service Workers(MDN)

Cache(MDN)

Service Worker使用方式

JavaScript Promise:簡介

Promise(MDN)

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

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

相關(guān)文章

  • PWA學(xué)習(xí)實踐】(5)在Web中進行服務(wù)端消息推送

    摘要:本文是學(xué)習(xí)與實踐系列的第五篇文章。實際上,消息推送與提醒是兩個功能和。在這一篇里,我們先來學(xué)習(xí)如何使用進行消息推送。而當服務(wù)端要推送消息時,會使用私鑰對發(fā)送的數(shù)據(jù)進行數(shù)字簽名,并根據(jù)數(shù)字簽名生成一個叫請求頭。 《PWA學(xué)習(xí)與實踐》系列文章已整理至gitbook - PWA學(xué)習(xí)手冊,文字內(nèi)容已同步至learning-pwa-ebook。轉(zhuǎn)載請注明作者與出處。 本文是《PWA學(xué)習(xí)與實踐》系...

    suemi 評論0 收藏0
  • PWA學(xué)習(xí)實踐】(2) 使用Manifest,你的WebApp更“Native”

    摘要:,不過在上會導(dǎo)致狀態(tài)欄不顯示任何東西。下面是項目中的相關(guān)設(shè)置圖書搜索中的處理方式與類似,中也有自己的標簽來指示相應(yīng)的資源。 《PWA學(xué)習(xí)與實踐》系列文章已整理至gitbook - PWA學(xué)習(xí)手冊,文字內(nèi)容已同步至learning-pwa-ebook。轉(zhuǎn)載請注明作者與出處。 本文是《PWA學(xué)習(xí)與實踐》系列的第二篇文章。文中的代碼都可以在learning-pwa的manifest分支上找到...

    flyer_dev 評論0 收藏0
  • PWA學(xué)習(xí)實踐】(1) 2018,開始你的PWA學(xué)習(xí)之旅

    摘要:學(xué)習(xí)與實踐系列文章已整理至學(xué)習(xí)手冊,文字內(nèi)容已同步至。本系列文章學(xué)習(xí)與實踐會逐步拆解背后的各項技術(shù),通過實例代碼來講解這些技術(shù)的應(yīng)用方式。而隨著在中也開始支持其中的某些技術(shù),的舞臺更大了。這個最開始是不具備任何的能力。 《PWA學(xué)習(xí)與實踐》系列文章已整理至gitbook - PWA學(xué)習(xí)手冊,文字內(nèi)容已同步至learning-pwa-ebook。轉(zhuǎn)載請注明作者與出處。 PWA作為今年最火...

    blastz 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<