欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

談?wù)劄槭裁茨愕?JavaScript 代碼如此冗長(zhǎng)

 更新時(shí)間:2019年01月30日 09:20:42   作者:前端大牛愛(ài)好者  
這篇文章主要介紹了談?wù)劄槭裁茨愕?JavaScript 代碼如此冗長(zhǎng),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧

又一年過(guò)去了,JavaScript發(fā)生了許多變化。但是,即使是2019年了,還是需要給一些幫助你編寫(xiě)干凈、整潔、有效、且具有擴(kuò)展性的代碼建議。

下面是讓你成為更好的開(kāi)發(fā)者的9條建議。

1. async / await

如果你還在為回調(diào)陷阱煩惱不已,那么就應(yīng)該趕快扔掉這些2014年的代碼了。除非絕對(duì)必要(比如某個(gè)庫(kù)要求回調(diào),或者出于性能的原因),否則不要再用回調(diào)了。Promise也不錯(cuò),但當(dāng)代碼規(guī)模越來(lái)越大時(shí),它們總是有些別扭。

我的解決方案就是async / await,能讓閱讀代碼變得更容易,代碼變得更整潔。實(shí)際上,Javascript中的任何Promise都可以await,只要你用的庫(kù)能返回Promise,就可以await它。實(shí)際上,async/await只不過(guò)是promise的語(yǔ)法糖而已。為了讓代碼正確運(yùn)行,你只需在函數(shù)前面加上async即可。

下面是個(gè)例子:

async function getData() {
  const result = await axios.get('https://dube.io/service/ping')
  const data = result.data

  console.log('data', data)

  return data
}

getData()

注意在頂層代碼是無(wú)法await的,await只能在async函數(shù)中使用。此外,async / await是在ES2017中引入的,所以務(wù)必要對(duì)代碼進(jìn)行編譯(transpile)。

2. 異步控制流

許多時(shí)候需要獲取多個(gè)數(shù)據(jù)集并在每個(gè)數(shù)據(jù)集上做一些處理,或者在所有異步調(diào)用都返回之后執(zhí)行某項(xiàng)任務(wù)。

for...of

假設(shè)網(wǎng)頁(yè)上有一些精靈寶可夢(mèng),我們需要獲取每一只的詳細(xì)信息。我們不能等待所有調(diào)用結(jié)束,因?yàn)槲覀儾恢酪还灿卸嗌僦?。我們希望能在獲取一部分?jǐn)?shù)據(jù)之后立即更新數(shù)據(jù)集,這時(shí)候就可以使用for...of在一個(gè)數(shù)組上進(jìn)行循環(huán),然后在內(nèi)部加入async的代碼塊,但這樣做會(huì)造成阻塞,直到所有調(diào)用結(jié)束。一定要注意,這樣做有可能會(huì)造成性能瓶頸,但這樣做也不失為一種辦法。

例子如下:

import axios from 'axios'

let myData = [{id: 0}, {id: 1}, {id: 2}, {id: 3}]

async function fetchData(dataSet) {
  for(entry of dataSet) {
    const result = await axios.get(`https://ironhack-pokeapi.herokuapp.com/pokemon/${entry.id}`)
    const newData = result.data
    updateData(newData)

    console.log(myData)
  }
}

function updateData(newData) {
  myData = myData.map(el => {
    if(el.id === newData.id) return newData
    return el
  })
}

fetchData(myData)

這些例子實(shí)際上都能運(yùn)行,可以自行復(fù)制粘貼到你喜歡的代碼沙盒工具中。

Promise.all

怎樣才能并行獲取所有寶可夢(mèng)呢?我們可以await所有的promise,只需用Promise.all即可:

import axios from 'axios' 

let myData = [{id: 0}, {id: 1}, {id: 2}, {id: 3}]

async function fetchData(dataSet) {
  const pokemonPromises = dataSet.map(entry => {
    return axios.get(`https://ironhack-pokeapi.herokuapp.com/pokemon/${entry.id}`)
  })

  const results = await Promise.all(pokemonPromises)

  results.forEach(result => {
    updateData(result.data)
  })

  console.log(myData) 
}

function updateData(newData) {
  myData = myData.map(el => {
    if(el.id === newData.id) return newData
    return el
  })
}

fetchData(myData)

for...of和Promise.all都是在ES6+中引用的,所以代碼需要編譯。

3. 解構(gòu)和默認(rèn)值

我們現(xiàn)在回到前面的例子:

const result = axios.get(`https://ironhack-pokeapi.herokuapp.com/pokemon/${entry.id}`)
const data = result.data

這段代碼有個(gè)更簡(jiǎn)單的寫(xiě)法。我們可以使用解構(gòu)來(lái)從一個(gè)數(shù)組或?qū)ο笾蝎@取一個(gè)或多個(gè)值??梢赃@樣寫(xiě):

const { data } = await axios.get(...)

這樣就能節(jié)省一行代碼!還可以進(jìn)行重命名:

const { data: newData } = await axios.get(...)

另一個(gè)小技巧就是在解構(gòu)時(shí)制定默認(rèn)值。這樣能保證變量永遠(yuǎn)不會(huì)為undefine,因此就不需要手工檢查變量了。

const { id = 5 } = {}
console.log(id) // 5

這些技巧也可以用在函數(shù)參數(shù)上,例如:

function calculate({operands = [1, 2], type = 'addition'} = {}) {
  return operands.reduce((acc, val) => {
    switch(type) {
      case 'addition':
        return acc + val
      case 'subtraction':
        return acc - val
      case 'multiplication':
        return acc * val
      case 'division':
        return acc / val
    }
  }, ['addition', 'subtraction'].includes(type) ? 0 : 1)
}

console.log(calculate()) // 3
console.log(calculate({type: 'division'})) // 0.5
console.log(calculate({operands: [2, 3, 4], type: 'multiplication'})) // 24

第一眼看上去這個(gè)例子可能不太容易理解,但多花些時(shí)間研究下是有好處的。當(dāng)我們不給函數(shù)傳遞參數(shù)時(shí),就會(huì)使用默認(rèn)值。如果給函數(shù)傳遞參數(shù),那么不存在的參數(shù)就會(huì)使用默認(rèn)值。

解構(gòu)和默認(rèn)值是在ES6+中引入的,所以代碼需要編譯。

4. 真值和假值

在使用默認(rèn)值時(shí),經(jīng)常需要檢查存在的值。但是,你還可以直接使用真值和假值。這樣能改善代碼并節(jié)省好多字符,使代碼更加流暢。我經(jīng)??吹饺藗冞@樣寫(xiě):

if(myBool === true) {
 console.log(...)
}
// OR
if(myString.length > 0) {
 console.log(...)
}
// OR
if(isNaN(myNumber)) {
 console.log(...)
}

這些代碼可以縮寫(xiě)成:

if(myBool) {
 console.log(...)
}
// OR
if(myString) {
 console.log(...)
}
// OR
if(!myNumber) {
 console.log(...)
}

要真正理解這些語(yǔ)句的好處,你必須要理解真值和假值都是什么。下面是部分摘要:

假值

  • 長(zhǎng)度為0的字符串
  • 數(shù)字0
  • false
  • undefined
  • null
  • NaN

真值

  • 空數(shù)組
  • 空對(duì)象
  • 任何其他東西

當(dāng)檢查真值或假值時(shí),不需要明確寫(xiě)出比較,這相當(dāng)于使用雙等號(hào) == 而不是三等號(hào) ===。一般來(lái)說(shuō),這種用法的行為與預(yù)想是一致的,但有可能會(huì)遇到bug。比如,我最常遇到但就是有關(guān)數(shù)字0的bug。

5. 邏輯運(yùn)算符和三元運(yùn)算符

這些運(yùn)算符也是用來(lái)縮減代碼的,節(jié)省下寶貴的代碼行數(shù)。經(jīng)常有許多工具可以保持代碼干凈整潔,但這些工具也會(huì)造成混亂,特別是在改變它們時(shí)。

邏輯運(yùn)算符

邏輯運(yùn)算符可以組合兩個(gè)表達(dá)式,并返回true或false,或者匹配的值。常用的有&&,意思是“與”,還有 || 意思是“或”。我們來(lái)看看:

console.log(true && true) // true
console.log(false && true) // false
console.log(true && false) // false
console.log(false && false) // false
console.log(true || true) // true
console.log(true || false) // true
console.log(false || true) // true
console.log(false || false) // false

根據(jù)上一部分關(guān)于真值和假值的知識(shí),我們可以將邏輯運(yùn)算符組合起來(lái)。在使用邏輯運(yùn)算符時(shí),會(huì)使用以下規(guī)則:

  • && :返回第一個(gè)值為假的表達(dá)式的值。如果不存在,則返回最后一個(gè)值為真的值。
  • || :返回第一個(gè)值為假的表達(dá)式的值。如果不存在,則返回最后一個(gè)值為假的值。
console.log(0 && {a: 1}) // 0
console.log(false && 'a') // false
console.log('2' && 5) // 5
console.log([] || false) // []
console.log(NaN || null) // null
console.log(true || 'a') // true

三元運(yùn)算符

三元運(yùn)算符很像邏輯表達(dá)式,但它由三個(gè)部分組成:

  • 比較部分,返回假值或真值;
  • 第一個(gè)值,如果比較為真;
  • 第二個(gè)值,如果比較為假。

下面是例子:

const lang = 'German'
console.log(lang === 'German' ? 'Hallo' : 'Hello') // Hallo
console.log(lang ? 'Ja' : 'Yes') // Ja
console.log(lang === 'French' ? 'Bon soir' : 'Good evening') // Good evening

6. 鏈?zhǔn)讲僮?/strong>

你遇到過(guò)這個(gè)問(wèn)題嗎?在訪(fǎng)問(wèn)嵌套對(duì)象的屬性時(shí),無(wú)法事先確定對(duì)象的屬性是否存在?可能不得不寫(xiě)這樣的代碼:

let data
if(myObj && myObj.firstProp && myObj.firstProp.secondProp && myObj.firstProp.secondProp.actualData) data = myObj.firstProp.secondProp.actualData

這段代碼很荒謬,我們還有更好的辦法,至少是在建議中的辦法(下面說(shuō)了怎樣啟用該辦法)。這個(gè)辦法稱(chēng)為optional chaining,用法如下:

const data = myObj?.firstProp?.secondProp?.actualData

用這個(gè)方法檢查嵌套屬性非常流暢,代碼也能變得更干凈。

目前,optional chaining還不是官方標(biāo)準(zhǔn)的一部分,但它是個(gè)stage-1的實(shí)驗(yàn)性功能。需要在babelrc中加入@babel/plugin-proposal-optional-chaining來(lái)啟用它。

7. 類(lèi)屬性和綁定

JavaScript中的函數(shù)綁定是個(gè)非常常見(jiàn)的任務(wù)。由于ES6標(biāo)準(zhǔn)引入了箭頭函數(shù),我們現(xiàn)在可以自動(dòng)地用定義的形式綁定函數(shù)——這方法非常好用,現(xiàn)在的JavaScript開(kāi)發(fā)者都在用它。之前類(lèi)剛剛出現(xiàn)時(shí)是沒(méi)辦法使用箭頭函數(shù)的,因?yàn)轭?lèi)需要用某種特殊的方式來(lái)定義。我們需要在某個(gè)地方進(jìn)行綁定,例如在構(gòu)造函數(shù)里(在React.js中最好這樣做)。

我很討厭需要先定義類(lèi)方法再綁定方法的流程,不過(guò)現(xiàn)在可以通過(guò)箭頭函數(shù)進(jìn)行自動(dòng)綁定。箭頭函數(shù)現(xiàn)在可以直接在類(lèi)中使用。

下面是個(gè)例子,其中的_increaseCount被綁定了:

class Counter extends React.Component {
  constructor(props) {
    super(props)
    this.state = { count: 0 }
  }

  render() {
    return(
      <div>
        <h1>{this.state.count}</h1> 
        <button onClick={this._increaseCount}>Increase Count</button>
      </div>
    )
  }

  _increaseCount = () => {
    this.setState({ count: this.state.count + 1 })
  }
}

目前,類(lèi)屬性不是官方標(biāo)準(zhǔn)的一部分,但是個(gè)stage-3的實(shí)驗(yàn)性功能。必須在babelrc中添加@babel/plugin-proposal-class-properties才能使用它。

8. 使用parcel

作為前端開(kāi)發(fā)者,你肯定會(huì)遇到打包和編譯代碼的問(wèn)題。

長(zhǎng)時(shí)間以來(lái),實(shí)踐中的標(biāo)準(zhǔn)是webpack。我最初用的是webpack版本1,當(dāng)時(shí)用起來(lái)很痛苦,需要不斷修改嘗試各種配置選項(xiàng),我在上面花了無(wú)數(shù)個(gè)小時(shí)想辦法讓它工作。一旦弄好我就絕不會(huì)再碰它,以免不小心破壞什么。幾個(gè)月之后我遇到了parcel,總算松了口氣。它幾乎可以不加任何配置拿來(lái)即用,但你依然可以在需要的時(shí)候進(jìn)行改變。它還支持插件,類(lèi)似于webpack和babel,但非???。

如果你不知道parcel,我建議你一定要試試。

9. 自己寫(xiě)更多代碼

這一條很有意思,這個(gè)話(huà)題我已經(jīng)討論過(guò)很多次了。

即使是CSS,許多人也喜歡用現(xiàn)成的庫(kù),比如bootstrap。至于JavaScript,現(xiàn)在還有很多人在用jQuery以及各種小型庫(kù)進(jìn)行表單驗(yàn)證、跑馬燈等等。雖然使用庫(kù)天經(jīng)地義,但我強(qiáng)烈建議你自己寫(xiě)更多的代碼,而不是依賴(lài)于安裝各種npm包。當(dāng)然,大型的庫(kù)(甚至框架)需要整個(gè)團(tuán)隊(duì)去構(gòu)建,如moment.js或react-dateicker,自己寫(xiě)是不現(xiàn)實(shí)的。

但是,其他的大部分東西都可以自己寫(xiě)。這樣能帶來(lái)三個(gè)好處:

你清楚地知道代碼的內(nèi)容;

在某個(gè)點(diǎn)上你開(kāi)始真正理解編程,知道內(nèi)部的工作原理;

可以防止代碼膨脹。

最初直接使用npm包很方便。自己實(shí)現(xiàn)一些功能會(huì)花很多時(shí)間。但是,如果安裝的包并不能正常工作,而需要換別的方法,就得花更多的時(shí)間去閱讀其API。而在自己實(shí)現(xiàn)時(shí),你可以為項(xiàng)目100%地量身定做。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • 使用layui日期控件laydate對(duì)開(kāi)始和結(jié)束時(shí)間進(jìn)行聯(lián)動(dòng)控制的方法

    使用layui日期控件laydate對(duì)開(kāi)始和結(jié)束時(shí)間進(jìn)行聯(lián)動(dòng)控制的方法

    今天小編就為大家分享一篇使用layui日期控件laydate對(duì)開(kāi)始和結(jié)束時(shí)間進(jìn)行聯(lián)動(dòng)控制的方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-09-09
  • 微信小程序?qū)崿F(xiàn)手風(fēng)琴折疊面板

    微信小程序?qū)崿F(xiàn)手風(fēng)琴折疊面板

    這篇文章主要為大家詳細(xì)介紹了微信小程序?qū)崿F(xiàn)手風(fēng)琴折疊面板,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-05-05
  • javascript 打印內(nèi)容方法小結(jié)

    javascript 打印內(nèi)容方法小結(jié)

    關(guān)于JS打印問(wèn)題!網(wǎng)上收集+自己總結(jié) ,需要的朋友可以參考下。
    2009-11-11
  • 如何使用50行javaScript代碼實(shí)現(xiàn)簡(jiǎn)單版的call,apply,bind

    如何使用50行javaScript代碼實(shí)現(xiàn)簡(jiǎn)單版的call,apply,bind

    這篇文章主要介紹了50行javaScript代碼實(shí)現(xiàn)簡(jiǎn)單版的call,apply,bind過(guò)程詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-08-08
  • js+springMVC 提交數(shù)組數(shù)據(jù)到后臺(tái)的實(shí)例

    js+springMVC 提交數(shù)組數(shù)據(jù)到后臺(tái)的實(shí)例

    今天小編就為大家分享一篇js+springMVC 提交數(shù)組數(shù)據(jù)到后臺(tái)的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2019-09-09
  • bootstrap輸入框組件使用方法詳解

    bootstrap輸入框組件使用方法詳解

    這篇文章主要為大家詳細(xì)介紹了bootstrap輸入框組件使用方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2017-01-01
  • javascript定時(shí)器的簡(jiǎn)單應(yīng)用示例【控制方塊移動(dòng)】

    javascript定時(shí)器的簡(jiǎn)單應(yīng)用示例【控制方塊移動(dòng)】

    這篇文章主要介紹了javascript定時(shí)器的簡(jiǎn)單應(yīng)用,結(jié)合javascript事件觸發(fā)控制方塊移動(dòng)操作分析了javascript定時(shí)器使用相關(guān)操作技巧,需要的朋友可以參考下
    2019-06-06
  • 最新評(píng)論