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

資訊專欄INFORMATION COLUMN

ES8初探

Shisui / 2582人閱讀

ES8

在es8中主要有6個(gè)特性:
主要的有:

Shared memory and atomics (共享內(nèi)存和原子)

Async Functions(異步函數(shù))

其他的特性:

Object.values/Object.entries (配合Object.keys使用)

String padding (字符串填充)

Object.getOwnPropertyDescriptors()

Trailing commas in function parameter lists and calls(在函數(shù)參數(shù)列表和調(diào)用中減少逗號(hào)的使用)

首先我們來(lái)介紹一下主要的兩個(gè)特性:

Shared memory and atomics (共享內(nèi)存和原子)

在我們先需要主要需要了解SharedArrayBuffer 和 Atomics

SharedArrayBuffer

在了解SharedArrayBuffer 之前我們需要了解下需要share的這個(gè)ArrayBuffer:

ArrayBuffer對(duì)象代表儲(chǔ)存二進(jìn)制數(shù)據(jù)的一段內(nèi)存,它不能直接讀寫(xiě),只能通過(guò)視圖(TypedArray視圖和DataView視圖)來(lái)讀寫(xiě),視圖的作用是以指定格式解讀二進(jìn)制數(shù)據(jù)。這個(gè)接口的原始設(shè)計(jì)目的,與 WebGL 項(xiàng)目有關(guān)。所謂WebGL,就是指瀏覽器與顯卡之間的通信接口,為了滿足 JavaScript 與顯卡之間大量的、實(shí)時(shí)的數(shù)據(jù)交換,它們之間的數(shù)據(jù)通信必須是二進(jìn)制的,而不能是傳統(tǒng)的文本格式。文本格式傳遞一個(gè)32位整數(shù),兩端的 JavaScript 腳本與顯卡都要進(jìn)行格式轉(zhuǎn)化,將非常耗時(shí)。這時(shí)要是存在一種機(jī)制,可以像 C 語(yǔ)言那樣,直接操作字節(jié),將4個(gè)字節(jié)的32位整數(shù),以二進(jìn)制形式原封不動(dòng)地送入顯卡,腳本的性能就會(huì)大幅提升。

那么我們?yōu)槭裁葱枰猄haredArrayBuffer這個(gè)function呢?
任何能夠從主線程負(fù)載減少工作的方法都對(duì)代碼運(yùn)行效率有幫助,某些情況下,ArrayBuffers 可以減少大量應(yīng)該由主線程做的工作(免去了格式轉(zhuǎn)換的耗時(shí)和壓力直接操作字符),但是也有些時(shí)候減少主線程負(fù)載是遠(yuǎn)遠(yuǎn)不夠的,有時(shí)你需要增援,你需要分割你的任務(wù),那么我們這個(gè)時(shí)候就需要傳說(shuō)中的多線程。
在 JavaScript 里,你可以借助 web worker 做這種事,但是web worker是不能共享內(nèi)存的,那么這意味著如果你想分配你的任務(wù)給別的線程,你需要完整把任務(wù)復(fù)制過(guò)去,這可以通過(guò) postMessage 實(shí)現(xiàn),postMessage 把你傳給它的任何對(duì)象都序列化,發(fā)送到其它 web worker,然后那邊接收后反序列化并放進(jìn)內(nèi)存??梢?jiàn)這個(gè)過(guò)程也是相當(dāng)慢的。我們下面來(lái)看下這個(gè)過(guò)程的code [from MDN]:

main.js

var first = document.querySelector("#number1");
var second = document.querySelector("#number2");

var result = document.querySelector(".result");

if (window.Worker) { // Check if Browser supports the Worker api.
    // Requires script name as input
    var myWorker = new Worker("worker.js");

// onkeyup could be used instead of onchange if you wanted to update the answer every time
// an entered value is changed, and you don"t want to have to unfocus the field to update its .value

    first.onchange = function() {
      myWorker.postMessage([first.value,second.value]); // Sending message as an array to the worker
      console.log("Message posted to worker");
    };

    second.onchange = function() {
      myWorker.postMessage([first.value,second.value]);
      console.log("Message posted to worker");
    };

    myWorker.onmessage = function(e) {
        result.textContent = e.data;
        console.log("Message received from worker");
    };
}

worker.js

onmessage = function(e) {
  console.log("Message received from main script");
  var workerResult = "Result: " + (e.data[0] * e.data[1]);
  console.log("Posting message back to main script");
  postMessage(workerResult);
}

從上面的code可以看出這樣傳來(lái)傳去是非常慢的。
那么我們想多個(gè) web worker 就可以同時(shí)讀寫(xiě)同一塊內(nèi)存,這樣我們就不用傳來(lái)傳去的了,這就是SharedArrayBuffers 為我們提供的。
我們來(lái)看下使用使用shareArrayBuffers是怎樣實(shí)現(xiàn)這個(gè)乘法的

main.js

"use strict";
var first = document.querySelector("#number1");
var second = document.querySelector("#number2");

var result = document.querySelector(".result");

if (window.Worker) { // Check if Browser supports the Worker api.
    // Requires script name as input
    var myWorker = new Worker("worker.js");

// onkeyup could be used instead of onchange if you wanted to update the answer every time
// an entered value is changed, and you don"t want to have to unfocus the field to update its .value
    var sharedBuffer = new SharedArrayBuffer(Int32Array.BYTES_PER_ELEMENT);
    first.onchange = function() {
        addNumber();
    }
      
    second.onchange = function() {
       addNumber();
    };
    const addNumber = () => {
        myWorker.postMessage({
            aTopic: [first.value, second.value],
            aBuf: sharedBuffer // The array buffer that we passed to the transferrable section 3 lines below
        });
      console.log("Message posted to worker");
      result.textContent = new Int32Array(sharedBuffer)[0];
      console.log("Message received from worker");
    }
}

worker.js

onmessage = function(e) {
  console.log("Message received from main script");
  var workerResult = e.data.aTopic[0] * e.data.aTopic[1];
  new Int32Array(e.data.aBuf)[0] = workerResult;
}

我們這個(gè)時(shí)候也不用擔(dān)心postMessage 伴有時(shí)延的通信。但是多個(gè)worker同時(shí)訪問(wèn)一塊內(nèi)存這個(gè)時(shí)候會(huì)出現(xiàn)競(jìng)爭(zhēng)的問(wèn)題的。
從 CPU 層面看,增加一個(gè)變量值需要三條指令,這是因?yàn)橛?jì)算機(jī)同時(shí)有長(zhǎng)期存儲(chǔ)器(內(nèi)存)和短期存儲(chǔ)器(寄存器所有的線程共享同一個(gè)長(zhǎng)期存儲(chǔ)器(內(nèi)存),但是短期存儲(chǔ)器(寄存器)并不是共享的。每個(gè)線程需要把值先從內(nèi)存搬到寄存器,之后就可以在寄存器上進(jìn)行計(jì)算了,再然后會(huì)把計(jì)算后的值寫(xiě)回內(nèi)存)。

這個(gè)因?yàn)楦?jìng)爭(zhēng)的關(guān)系會(huì)產(chǎn)生錯(cuò)誤的結(jié)果,那么我們應(yīng)該怎樣讓SharedArrayBuffers 發(fā)揮他應(yīng)有的價(jià)值呢?
這個(gè)時(shí)候伴隨它誕生的還有:

Atomics

原子操作做的一件事就是在多線程中讓計(jì)算機(jī)按照人所想的單操作方式工作。
這就是為什么被叫做原子操作,因?yàn)樗梢宰屢粋€(gè)包含多條指令(指令可以暫停和恢復(fù))的操作執(zhí)行起來(lái)像是一下子就完了,就好像一條指令,類似一個(gè)不可分割的原子。

var sab = new SharedArrayBuffer(1024);
var ta = new Uint8Array(sab);
//The static Atomics.add() method adds a given value at a given //position in the array and returns the old value at that position.
Atomics.add(ta, 0, 1); // returns 0, the old value
Atomics.load(ta, 0); // 1
async 函數(shù)

首先我們來(lái)看下在async 函數(shù)誕生之前我們是怎么處理異步的,我先簡(jiǎn)單列舉幾個(gè)編程模型:
回調(diào)函數(shù):

// more code
function loading(callback) {
    // wait 3s
    setTimeout(function () {
        callback();
    }, 3000);
}
function show() {
    // show the data.
}
loading(show);
// more code

在這種情況下容易出現(xiàn)大家所熟知的回調(diào)黑洞:

A(function () {
    B(function () {
        C(function() {
            D(function() {
                // ...
            })
        })
    })
})

Promise模式
所謂 Promise,就是一個(gè)對(duì)象,用來(lái)傳遞異步操作的消息。它代表了某個(gè)未來(lái)才會(huì)知道結(jié)果的事件(通常是一個(gè)異步操作),并且這個(gè)事件提供統(tǒng)一的 API,可供進(jìn)一步處理。

let promise = new Promise(function(resolve, reject) {
  console.log("Promise");
  resolve();
});

promise.then(function() {
  console.log("Resolved.");
});

console.log("Hi!");

// Promise
// Hi!
// Resolved

這種實(shí)現(xiàn)方式和AngularJs中的Promise用法相近。
下面簡(jiǎn)單提下ES6中的Generator 函數(shù)的使用:

function* gen(x) {
  var y = yield x + 2;
  return y;
}

var g = gen(1);
g.next() // { value: 3, done: false }
g.next() // { value: undefined, done: true }

gen函數(shù)返回不是函數(shù)運(yùn)行結(jié)果,而是一個(gè)指向內(nèi)部狀態(tài)的指針對(duì)象,也就是遍歷器對(duì)象(Iterator Object)。
其實(shí)在ES8之前我們還有許多其他的異步實(shí)現(xiàn)方式,比如:
在angualrJs中我們實(shí)現(xiàn)的subscribe, unsubscribe, publish這種消息訂閱/發(fā)布模式其實(shí)也是實(shí)現(xiàn)異步的一種方式。
下面我們來(lái)說(shuō)說(shuō)今天的主角:async 函數(shù)
關(guān)鍵詞:await async
那么我們先看下await, await后面可以是Promise 對(duì)象和原始類型的值(數(shù)值、字符串和布爾值,但這時(shí)等同于同步操作),當(dāng)函數(shù)執(zhí)行的時(shí)候,一旦遇到await就會(huì)先返回,等到異步操作完成,再接著執(zhí)行函數(shù)體內(nèi)后面的語(yǔ)句。
async函數(shù)返回一個(gè) Promise 對(duì)象,可以使用then方法添加回調(diào)函數(shù)。
我們看下下面的代碼來(lái)理解上面的話:

function resolveAfter2Seconds(x) {
  return new Promise(resolve => {
    setTimeout(() => {
      resolve(x);
    }, 2000);
  });
}

async function add(x) {
  var a = await resolveAfter2Seconds(20);
  var b = await resolveAfter2Seconds(30);
  return x + a + b;
}

add(10).then(v => {
  console.log(v);  // prints 60 after 4 seconds.
});
Object.values/Object.entries

ES5 引入了Object.keys方法,返回一個(gè)數(shù)組,成員是參數(shù)對(duì)象自身的所有可遍歷(enumerable)屬性的鍵名。

var obj = { foo: "bar", baz: 42 };
Object.keys(obj)
// ["foo", "baz"]

es8中新添加的Object.values/Object.entries作為遍歷對(duì)象的一種補(bǔ)充手段。
Object.values方法返回一個(gè)數(shù)組,成員是參數(shù)對(duì)象自身的所有可遍歷(enumerable)屬性的鍵值。

var obj = { 100: "a", "a": "b", 7: "c" };
Object.values(obj)
// ["c", "a", "b"]

Object.entries方法返回一個(gè)數(shù)組,成員是參數(shù)對(duì)象自身的所有可遍歷(enumerable)屬性的鍵值對(duì)數(shù)組。

var obj = { "foo": "bar", "baz": 42 , "1": 43};
Object.entries(obj)
// [["1", 43], ["foo", "bar"], ["baz", 42] ]

以上的遍歷對(duì)象的屬性,都遵守同樣的屬性遍歷的次序規(guī)則。
首先遍歷所有屬性名為數(shù)值的屬性,按照數(shù)字排序。
其次遍歷所有屬性名為字符串的屬性,按照生成時(shí)間排序。
最后遍歷所有屬性名為 Symbol 值的屬性,按照生成時(shí)間排序。
(Object.keys,Object.values,Object.entries都會(huì)過(guò)濾屬性名為 Symbol(一種新的原始數(shù)據(jù)類型, 表示獨(dú)一無(wú)二的值) 值的屬性。)

String padding

字符填充函數(shù),在es8中引入的兩個(gè)方法: String.padStart 和String.padEnd, 這兩個(gè)放發(fā)的主要是為了在一定程度上填充字符串的長(zhǎng)度, 語(yǔ)法如下:

str.padStart(targetLength [, padString])

str.padEnd(targetLength [, padString])

這個(gè)方法主要是為了實(shí)現(xiàn)字符串補(bǔ)全長(zhǎng)度的功能。如果某個(gè)字符串不夠指定長(zhǎng)度,會(huì)在頭部或尾部補(bǔ)全。padStart()用于頭部補(bǔ)全,padEnd()用于尾部補(bǔ)全。

"x".padStart(5, "ab") // "ababx"
"x".padStart(4, "ab") // "abax"
"x".padEnd(5, "ab") // "xabab"
"x".padEnd(4, "ab") // "xaba"

//如果原字符串的長(zhǎng)度,等于或大于指定的最小長(zhǎng)度,則返回原字符串。
"xxx".padStart(2, "ab") // "xxx"
"xxx".padEnd(2, "ab") // "xxx"

//如果省略第二個(gè)參數(shù),默認(rèn)使用空格補(bǔ)全長(zhǎng)度。
"x".padStart(4) // "   x"
"x".padEnd(4) // "x   "

//用途: 為數(shù)值補(bǔ)全指定位數(shù)。
"1".padStart(10, "0") // "0000000001"
"12".padStart(10, "0") // "0000000012"
"123456".padStart(10, "0") // "0000123456"

//另一個(gè)用途是提示字符串格式。
"12".padStart(10, "YYYY-MM-DD") // "YYYY-MM-12"
"09-12".padStart(10, "YYYY-MM-DD") // "YYYY-09-12"
getOwnPropertyDescriptors函數(shù)

getOwnPropertyDescriptors函數(shù): 返回指定對(duì)象所有自身屬性(非繼承屬性)的描述對(duì)象。
ES5 有一個(gè)Object.getOwnPropertyDescriptor方法,返回某個(gè)對(duì)象屬性的描述對(duì)象(descriptor)。

var obj = { p: "a" };

Object.getOwnPropertyDescriptor(obj, "p")
// Object { value: "a",
//   writable: true,
//   enumerable: true,
//   configurable: true
// }

ES2017 引入了Object.getOwnPropertyDescriptors方法,返回指定對(duì)象所有自身屬性(非繼承屬性)的描述對(duì)象。

const obj = {
  foo: 123,
  get bar() { return "abc" }
};

Object.getOwnPropertyDescriptors(obj)
// { foo:
//    { value: 123,
//      writable: true,
//      enumerable: true,
//      configurable: true },
//   bar:
//    { get: [Function: bar],
//      set: undefined,
//      enumerable: true,
//      configurable: true } }

value

包含這個(gè)屬性的數(shù)據(jù)值。讀取屬性值的時(shí)候,從這個(gè)位置讀;寫(xiě)入屬性值的時(shí)候,把新值保存在這個(gè)位置。該特性的默認(rèn)值為undefined。直接在對(duì)象上定義的屬性,該特性被設(shè)置為指定的值

writable

表示能否修改屬性的值。直接在對(duì)象上定義的屬性,該特性默認(rèn)為true

get

獲取該屬性的訪問(wèn)器函數(shù)(getter)。如果沒(méi)有訪問(wèn)器, 該值為undefined。(僅針對(duì)包含訪問(wèn)器或設(shè)置器的屬性描述有效)

set

獲取該屬性的設(shè)置器函數(shù)(setter)。 如果沒(méi)有設(shè)置器, 該值為undefined。(僅針對(duì)包含訪問(wèn)器或設(shè)置器的屬性描述有效)

configurable

表示能否通過(guò)delete刪除屬性從而重新定義屬性。,能否修改屬性的特性,或者能否把屬性修改為訪問(wèn)器屬性。直接在對(duì)象上定義的屬性,該特性默認(rèn)為true;

enumerable

表示能否通過(guò)for-in循環(huán)返回屬性。直接在對(duì)象上定義的屬性,該特性默認(rèn)為true

應(yīng)用:淺拷貝

const shallowClone = (obj) => Object.create(
  Object.getPrototypeOf(obj),
  Object.getOwnPropertyDescriptors(obj)
);

在裝飾器上應(yīng)用

Trailing commas in function parameter lists and calls(結(jié)尾逗號(hào))

此處結(jié)尾逗號(hào)指的是在函數(shù)參數(shù)列表中最后一個(gè)參數(shù)之后的逗號(hào)以及函數(shù)調(diào)用時(shí)最后一個(gè)參數(shù)之后的逗號(hào)。ES8 允許在函數(shù)定義或者函數(shù)調(diào)用時(shí),最后一個(gè)參數(shù)之后存在一個(gè)結(jié)尾逗號(hào)而不報(bào) SyntaxError 的錯(cuò)誤。示例代碼如下:

函數(shù)聲明時(shí)

function es8(var1, var2, var3,) {
  // ...
}

函數(shù)調(diào)用時(shí)
es8(10, 20, 30,);
ES8的這項(xiàng)新特性受啟發(fā)于對(duì)象或者數(shù)組中最后一項(xiàng)內(nèi)容之后的逗號(hào),如 [10, 20, 30,] 和 { x: 1, } 。

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

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

相關(guān)文章

  • ECMAScript 2017(ES8)特性概述

    摘要:特性概述整理自,歸納于筆者的現(xiàn)代開(kāi)發(fā)語(yǔ)法基礎(chǔ)與實(shí)踐技巧系列文章中也歡迎關(guān)注前端每周清單系列獲得一手資訊。本部分則介紹了新的構(gòu)造器與包含靜態(tài)方法的命名空間對(duì)象。 ECMAScript 2017(ES8)特性概述 整理自 ES8 was Released and here are its Main New Features,歸納于筆者的現(xiàn)代 JavaScript 開(kāi)發(fā):語(yǔ)法基礎(chǔ)與實(shí)踐技巧系...

    Kyxy 評(píng)論0 收藏0
  • es7求冪運(yùn)算符和es8簡(jiǎn)單介紹

    摘要:求冪運(yùn)算符和簡(jiǎn)單介紹求冪運(yùn)算符求冪運(yùn)算符求冪運(yùn)算符打印打印打印打印打印判斷數(shù)組里面有沒(méi)有那個(gè)值在字符串前面填充打印打印打印打印打印在字符串后邊填充打印打印打印打印打印注釋其中第一個(gè)參數(shù)是目標(biāo)長(zhǎng)度,第二個(gè)參數(shù)是填充字符串,默認(rèn)的值是空格。 es7求冪運(yùn)算符和es8簡(jiǎn)單介紹 es7求冪運(yùn)算符: 求冪運(yùn)算符 ** operator (求冪運(yùn)算符) console.log(2**3); ...

    duan199226 評(píng)論0 收藏0
  • ES8 走馬觀花(ECMAScript2017 新特性)

    摘要:距離上一篇走馬觀花已經(jīng)快兩年時(shí)間了,上個(gè)月底正式發(fā)布,再寫(xiě)一篇姊妹篇,介紹新特性。會(huì)議的每一項(xiàng)決議必須大部分人贊同,并且沒(méi)有人強(qiáng)烈反對(duì)才可以通過(guò)。已經(jīng)準(zhǔn)備就緒,該特性會(huì)出現(xiàn)在年度發(fā)布的規(guī)范之中。 距離上一篇《ES6 走馬觀花》已經(jīng)快兩年時(shí)間了,上個(gè)月底 ES8 正式發(fā)布,再寫(xiě)一篇姊妹篇,介紹 ES8 新特性。 什么是 ES8 ES8 是 ECMA-262 標(biāo)準(zhǔn)第 8 版的簡(jiǎn)稱,從 ES...

    meislzhua 評(píng)論0 收藏0
  • 聊聊ES7與ES8特性

    摘要:我曾寫(xiě)過(guò)一篇關(guān)于博客個(gè)最佳特性,這次我打算聊聊和特性。自從年雙十一正式上線,累計(jì)處理了億錯(cuò)誤事件,得到了金山軟件百姓網(wǎng)等眾多知名用戶的認(rèn)可。 譯者按: 轉(zhuǎn)眼ES6發(fā)布2年了,是時(shí)候了解一下ES7與ES8特性了! 原文: ES7 and ES8 Features 譯者: Fundebug 為了保證可讀性,本文采用意譯而非直譯,并且對(duì)源代碼進(jìn)行了大量修改。另外,本文版權(quán)歸原作者所有...

    fxp 評(píng)論0 收藏0
  • 【重溫基礎(chǔ)】8.字符串

    摘要:字符串拓展在我們判斷字符串是否包含另一個(gè)字符串時(shí),之前,我們只有方法,之后我們又多了三種方法返回布爾值,表示是否找到參數(shù)字符串。返回布爾值,表示參數(shù)字符串是否在原字符串的頭部。 本文是 重溫基礎(chǔ) 系列文章的第八篇。今日感受:人在異鄉(xiāng),也不能忘記湯圓。 系列目錄: 【復(fù)習(xí)資料】ES6/ES7/ES8/ES9資料整理(個(gè)人整理) 【重溫基礎(chǔ)】1.語(yǔ)法和數(shù)據(jù)類型 【重溫基礎(chǔ)】2.流程控制和...

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

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

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<