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

資訊專(zhuān)欄INFORMATION COLUMN

通過(guò)一個(gè)場(chǎng)景實(shí)例 了解前端處理大數(shù)據(jù)的無(wú)限可能

willin / 991人閱讀

摘要:隨著前端的飛速發(fā)展,在瀏覽器端完成復(fù)雜的計(jì)算,支配并處理大量數(shù)據(jù)已經(jīng)屢見(jiàn)不鮮。作為該實(shí)例本身的數(shù)據(jù)層。后人常以阿喀琉斯之踵譬喻這樣一個(gè)道理即使是再?gòu)?qiáng)大的英雄,他也有致命的死穴或軟肋。

隨著前端的飛速發(fā)展,在瀏覽器端完成復(fù)雜的計(jì)算,支配并處理大量數(shù)據(jù)已經(jīng)屢見(jiàn)不鮮。那么,如何在最小化內(nèi)存消耗的前提下,高效優(yōu)雅地完成復(fù)雜場(chǎng)景的處理,越來(lái)越考驗(yàn)開(kāi)發(fā)者功力,也直接決定了程序的性能。

本文展現(xiàn)了一個(gè)完全在控制臺(tái)就能模擬體驗(yàn)的實(shí)例,通過(guò)一步步優(yōu)化,實(shí)現(xiàn)了生產(chǎn)并操控1000000(百萬(wàn)級(jí)別)個(gè)對(duì)象的場(chǎng)景。

導(dǎo)讀:這篇文章涉及到 javascript 中 數(shù)組各種操作、原型原型鏈、ES6、classes 繼承、設(shè)計(jì)模式、控制臺(tái)分析 等內(nèi)容。要求閱讀者具有 js 面向?qū)ο笤鷮?shí)的基礎(chǔ)知識(shí)。如果你是初級(jí)前端開(kāi)發(fā)者,很容易被較為復(fù)雜的邏輯繞的云里霧里,“從入門(mén)到放棄”,不過(guò)建議先收藏。如果你是“老司機(jī)”,本文提供的解決思路希望對(duì)你有所啟發(fā),拋磚引玉。

場(chǎng)景和初級(jí)感知

具體來(lái)說(shuō),我們需要一個(gè)構(gòu)造函數(shù),或者說(shuō)類(lèi)似 factory 模式,實(shí)例化1000000個(gè)以上對(duì)象實(shí)例。

先來(lái)感知一下具體實(shí)現(xiàn):

Step1

打開(kāi)你的瀏覽器控制臺(tái),仔細(xì)觀(guān)察并復(fù)制粘貼以下代碼,觸發(fā)執(zhí)行。

a = new Array(1e6).fill(0);

我們創(chuàng)建了一個(gè)長(zhǎng)度為1000000的數(shù)組,數(shù)組的每一項(xiàng)元素都為0。

Step2

在數(shù)組 a 的基礎(chǔ)上,再生產(chǎn)一個(gè)長(zhǎng)度為1000000的數(shù)組 b,數(shù)組的每一項(xiàng)元素都是一個(gè)普通 javascript object,擁有 id 屬性,并且其 id 屬性值為其在元素中的 index 值;

b = a.map((val, ix) => ({id: ix}))
Step3

接下來(lái),在 b 的基礎(chǔ)上,再生產(chǎn)一個(gè)長(zhǎng)度為1000000的數(shù)組 c ,類(lèi)似于
b,同時(shí)我們?cè)黾右恍┢渌鼘傩裕沟脭?shù)組元素對(duì)象更加復(fù)雜一些:

c = a.map((val, ix) => ({id: ix, shape: "square", size: 10.5, color: "green"}))

語(yǔ)義上,我們可以更直觀(guān)的理解:c 就是包含了1000000個(gè)元素的數(shù)組,每一項(xiàng)都是一個(gè)綠色的、size 為10.5的小方塊。

如果你按照指示做了下來(lái),控制臺(tái)上會(huì)有以下內(nèi)容:

深層探究

你也許會(huì)想,這么大的數(shù)據(jù)量,內(nèi)存占用會(huì)是什么樣的情況呢?

好,我來(lái)帶你看看,點(diǎn)擊控制臺(tái) Profiles,選擇 Take Shapshot。在Window->Window 目錄下,根據(jù)內(nèi)存進(jìn)行篩選,你會(huì)得到:

很明顯,我們看到:

a數(shù)組:8MB;

b數(shù)組:40MB;

c數(shù)組:64MB

也許在實(shí)際場(chǎng)景中,除了1000000個(gè)綠色的、size為10.5的小方塊,我們還需要很多不同顏色,不同 size 的形狀。之前,這樣“{{BANNED}}”的需求常見(jiàn)于游戲應(yīng)用中。但是現(xiàn)在,復(fù)雜項(xiàng)目中類(lèi)似場(chǎng)景,也許距離你并不遙遠(yuǎn)。

ES6 Classes處理需求

簡(jiǎn)單“熱身”之后,我們了解了實(shí)際需求。接下來(lái),我們考察一下 ES6 Classes 處理這個(gè)問(wèn)題的情況。請(qǐng)重新刷新瀏覽器 tab,復(fù)制執(zhí)行以下代碼。

class Shape {
    constructor (id, shape = "square", size = 10.5, color = "green") {
        this.x = x; //  坐標(biāo)x軸
        this.y = y; //  坐標(biāo)y軸
        Object.assign(this, {id, shape, size, color})
    }
}

a = new Array(1e6).fill(0);
b = a.map((val, ix) => new Shape(ix));

我們使用了ES6 Classes,并擴(kuò)充了每個(gè)形狀的坐標(biāo)信息。
此時(shí),再來(lái)看一下內(nèi)存占用情況:

很明顯,此時(shí) b 數(shù)組由1000000個(gè)形狀組成,占據(jù)內(nèi)存:80MB,超過(guò)了先前數(shù)組的內(nèi)存消耗。也許這并不出乎意料,此時(shí)的b數(shù)組畢竟又多了兩個(gè)屬性。

優(yōu)化設(shè)計(jì):Two-Headed Classes

我們先來(lái)分析一下上面的實(shí)現(xiàn),熟悉原型鏈、原型概念的同學(xué)也許會(huì)明白,之前的方案產(chǎn)生的實(shí)例,順著原型鏈上溯,具有三層原型屬性:

第一層屬性:[id, shape, size, color, x, y]; 這一層屬性的 hasOwnproperty 為 true; 屬性存在于實(shí)例本身。

第二層:[Shape]; 順著原型鏈上溯,這一層 instance.__proto__ === Constructor.prototype; ( proto 左右兩邊 __ 被編輯器吃掉了,請(qǐng)見(jiàn)諒,下同)

第三層:[Object]; 這一層: instance.__proto__.__proto__ === Object.prototype; 如果在向上追溯,就為 null 了。

這樣的情況下,實(shí)際業(yè)務(wù)數(shù)據(jù)層只有一層,即為第一層。

但是,請(qǐng)仔細(xì)思考,如果有大量的不同顏色,不同size,不同形狀的情況下。單一數(shù)據(jù)層,是難以滿(mǎn)足我們需求的。
我們需要,再添加一層數(shù)據(jù)層,構(gòu)成所謂的 Two-Headed Classes!同時(shí),還需要對(duì)于默認(rèn)的屬性,實(shí)現(xiàn)共享,以節(jié)省內(nèi)存的占用。

什么什么?沒(méi)聽(tīng)明白,那就請(qǐng)看具體操作吧。

### 如何實(shí)現(xiàn)?
我們可以使用 Object.create 方法,這樣使得生產(chǎn)得到的實(shí)例的
proto 指向 b 數(shù)組的元素,然后在最頂層設(shè)計(jì)一個(gè) id 屬性。

也許這樣說(shuō)過(guò)于晦澀,那就直接參考代碼吧,請(qǐng)注意,這是本篇文章最難以理解的地方,請(qǐng)務(wù)必仔細(xì)揣摩:

two = Object.create(b[0]); 
// two.__proto__ === b[0]
two.id = 1;

還記得 b 數(shù)組是什么嘛?參考上文,它由

b = a.map((val, ix) => new Shape(ix));

得到。

這樣子的話(huà),對(duì)于每一個(gè)實(shí)例,我們有如下關(guān)系:

第一層:[id]; 這一層實(shí)例的 hasOwnproperty 為 true;

第二層:[id, shape, size, color, x, y]; 這一層 instance.__proto__ === Constructor.prototype;

第三層:[Shape];

第四層:[Object]; 這一層的再頂層,就為null了。

我們將 Shape 的一個(gè)實(shí)例作為一個(gè)新的 object 的原型,并復(fù)寫(xiě)了 id 屬性,原有的 id 屬性將作為默認(rèn) id。

當(dāng)然,上邊的代碼只是“個(gè)案”,我們進(jìn)行“生產(chǎn)化”:

proto = new Shape(0);
function newTwoHeaded (ix) {
    const obj = Object.create(proto);
    obj.id = ix;
    return obj
}
c = a.map((val, ix) => newTwoHeaded(ix));

這么做多加入了一個(gè)數(shù)據(jù)層,那么有什么“收獲”呢?我們來(lái)看一下b和c的內(nèi)存占用情況吧:

這表明:我們從80MB的b,優(yōu)化得到了64MB的c!
原因當(dāng)然就在于雖然多加了一層原型結(jié)構(gòu),但是第二層變成了“共享”。

當(dāng)然,如果到這里你還沒(méi)有暈的話(huà),可能要問(wèn):那第二層諸如 shape, size, color 這些屬性變成共享的之后,存在互相干擾怎么破解呢?

好問(wèn)題,我先不解答,先給大家看一下最后的final product:

class ShapeMaker {
    constructor () {
        Object.assign(this, ShapeMaker.defaults())
    }
    static defaults () {
        return {
            id: null,
            x: 0,
            y: 0,
            shape: "square",
            size: 0.5,
            color: "red",
            strokeColor: "yellow",
            hidden: false,
            label: null,
            labelOffset: [0, 0],
            labelFont: "10px sans-serif",
            labelColor: "black"
        }
    }
    newShape (id, x, y) {
        const obj = Object.create(this);
        return Object.assign(obj, {id, x, y})
    }
    setDefault (name, value) {
        this[name] = value;
    }
    getDefault (name) {
        return this[name]
    }
}

在實(shí)例化的時(shí)候,我們便可以這樣使用:

shapeProto = new ShapreMaker();
d = a.map((val, ix) => shapeProto.newShape(ix, ix/10, -ix/10))

就像上面所說(shuō)的,初始化實(shí)例時(shí),我們初始化了 id, x, y 這么三個(gè)參數(shù)。作為該實(shí)例本身的數(shù)據(jù)層。這個(gè)實(shí)例的原型上,也有類(lèi)似的參數(shù),來(lái)保證默認(rèn)值。這些原型上的屬性,對(duì)于實(shí)例數(shù)組中的每個(gè)實(shí)例,都是共享的。

為了更好的對(duì)比,如果設(shè)計(jì)是這樣子:

function fatShape (id, x, y) {
    const a = new shapeMaker();
    return Object.assign(a, {id, x, y})
}
e = a.map((val, ix) => fatShape(ix, ix/10, -ix/10))

那么所有屬性無(wú)法共享,而是各自拷貝了一份。在內(nèi)存的占用上,將是我們給出方案的三倍之多!

阿喀琉斯之踵

阿喀琉斯,是凡人珀琉斯和美貌仙女忒提斯的寶貝兒子。忒提斯為了讓兒子煉成“金鐘罩”,在他剛出生時(shí)就將其倒提著浸進(jìn)冥河,遺憾的是,乖?xún)罕荒赣H捏住的腳后跟卻不慎露在水外,全身留下了惟一一處“死穴”。后來(lái),阿喀琉斯被帕里斯一箭射中了腳踝而死去。
后人常以“阿喀琉斯之踵”譬喻這樣一個(gè)道理:即使是再?gòu)?qiáng)大的英雄,他也有致命的死穴或軟肋。

就像我們剛才提的到解決方案一樣,也有一些“不足”。問(wèn)題其實(shí)在之前我也已經(jīng)拋出:“第二層諸如:shape, size, color 這些屬性變成共享的之后,存在互相干擾怎么破解呢?”

這個(gè)問(wèn)題的答案其實(shí)也隱藏在上面的代碼中,很簡(jiǎn)單,就是我們?cè)趯?shí)例的自身屬性上,進(jìn)行復(fù)寫(xiě),而避免更改原型上的屬性造成污染。

如果你看的云里霧里,不要緊,馬上看一下我下面的代碼說(shuō)明:

d.every((item) => item.shape === "square") // true

打印為 true,是因?yàn)?d 數(shù)組中的每個(gè)實(shí)例的 shape 屬性,都在原型上,且初始值都為"square";

現(xiàn)在我們調(diào)用 setDefault 方法,實(shí)現(xiàn)對(duì)默認(rèn) shape 的改寫(xiě)。

shapeProto.setDefault("shape", "circle");
d.every((item) => item.shape === "square"); // false

因?yàn)榇藭r(shí)所有實(shí)例的 shape 都在原型上,并共享這個(gè)原型。更改之后,我們有:

d.every((item) => item.shape === "circle"); // true

但是,我只想把第一個(gè)實(shí)例的 shape 設(shè)置為 triangle,其他的不變,該怎么辦呢?只需要在第一個(gè)實(shí)例上,增加一個(gè) shape 屬性,進(jìn)行重寫(xiě):

d[0].shape = "triangle";
d.every((item) => item.shape === "circle"); // false

好吧,嘗試完畢之后,我們?cè)谧兓貋?lái)。

d[0].shape = "circle";

這時(shí)候,自然有:

d.every((item) => item.shape === "circle"); // true

同時(shí),再折騰一下:

d[0].shape = "triangle";
d.every((item) => item.shape === "triangle"); // false

相信下面的也不難理解了:

shapeProto.setDefault("shape", "triangle");
d.every((item) => item.shape === "triangle"); // true

這種模式其實(shí)比單純使用ES6 Classes要靈活的多,同時(shí)也節(jié)省了內(nèi)存。所有的靜態(tài)屬性都是共享的,但是共享的靜態(tài)屬性又都是可變的,可復(fù)寫(xiě)的。

總結(jié)

這篇文章,我們?cè)陂_(kāi)頭部分了解到了在大量數(shù)據(jù)的情況下,內(nèi)存的占用是如何一步一步變的沉重。同時(shí),我們提供了一種,在傳統(tǒng)的
Classes 之上增加一個(gè)數(shù)據(jù)層的方法,有效地解決了這個(gè)問(wèn)題。解決方案充分利用了 Object.create 等手段。

當(dāng)然,理解這些內(nèi)容并不簡(jiǎn)單,需要讀者有比較扎實(shí)的 javascript 基礎(chǔ)。在您閱讀過(guò)程當(dāng)中,有任何問(wèn)題,歡迎與我討論。

內(nèi)容借鑒了Owen Densmore最新文章:Two Headed ES6 Classes!,喜歡看英文原版的同學(xué)可以直接戳鏈接。中文翻譯版并非直譯,進(jìn)行了較大幅度的講解和增刪。

Happy Coding!

PS:
作者Github倉(cāng)庫(kù) 和 知乎問(wèn)答鏈接
歡迎各種形式交流。

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

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

相關(guān)文章

  • 阿里云前端周刊 - 第 16 期

    摘要:或許,它還終將改變用戶(hù)對(duì)移動(dòng)的期待。通過(guò)一個(gè)場(chǎng)景實(shí)例了解前端處理大數(shù)據(jù)的無(wú)限可能隨著前端的飛速發(fā)展,在瀏覽器端完成復(fù)雜的計(jì)算,支配并處理大量數(shù)據(jù)已經(jīng)屢見(jiàn)不鮮。 推薦 1. 神經(jīng)網(wǎng)絡(luò)入門(mén) http://www.ruanyifeng.com/blo... 眼下最熱門(mén)的技術(shù),絕對(duì)是人工智能,人工智能的底層模型是神經(jīng)網(wǎng)絡(luò)(neural network)。許多復(fù)雜的應(yīng)用(比如模式識(shí)別、自動(dòng)控制)...

    waruqi 評(píng)論0 收藏0
  • 2017-07-15 前端日?qǐng)?bào)

    摘要:前端日?qǐng)?bào)精選從設(shè)計(jì)到源碼用強(qiáng)類(lèi)型語(yǔ)言增強(qiáng)通過(guò)一個(gè)場(chǎng)景實(shí)例了解前端處理大數(shù)據(jù)的無(wú)限可能專(zhuān)題之從零實(shí)現(xiàn)的表單驗(yàn)證第一部分使用和技巧對(duì)表單進(jìn)行約束驗(yàn)證中文譯即將到來(lái)的正則表達(dá)式新特性掘金個(gè)快速編程技巧個(gè)人文章周刊第期相信控制像一樣使用 2017-07-15 前端日?qǐng)?bào) 精選 Redux從設(shè)計(jì)到源碼用強(qiáng)類(lèi)型語(yǔ)言GraphQL增強(qiáng)React通過(guò)一個(gè)場(chǎng)景實(shí)例 了解前端處理大數(shù)據(jù)的無(wú)限可能JavaSc...

    Vicky 評(píng)論0 收藏0
  • PHP小知識(shí)點(diǎn)

    摘要:那些瑣碎的知識(shí)點(diǎn)作者記錄的的很奇特很難記的知識(shí)點(diǎn)。易錯(cuò)知識(shí)點(diǎn)整理注意和的區(qū)別中和都是輸出的作用,但是兩者之間還是有細(xì)微的差別。今天手頭不忙,總結(jié)一下,分享過(guò)程中掌握的知識(shí)點(diǎn)。 深入理解 PHP 之:Nginx 與 FPM 的工作機(jī)制 這篇文章從 Nginx 與 FPM 的工作機(jī)制出發(fā),探討配置背后的原理,讓我們真正理解 Nginx 與 PHP 是如何協(xié)同工作的。 PHP 那些瑣碎的知識(shí)...

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

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

0條評(píng)論

閱讀需要支付1元查看
<