學會了ES6,就不會寫出那樣的程式碼

BrownHu發表於2018-04-09

宣告變數的新姿勢

用let不用var

ES6之前我們用var宣告一個變數,但是它有很多弊病:

  • 因為沒有塊級作用域,很容易宣告全域性變數
  • 變數提升
  • 可以重複宣告 還記得這道面試題嗎?
var a = [];
for (var i = 0; i < 10; i++) {
  a[i] = function () {
    console.log(i);
  };
}
a[6](); // 10
a[7](); // 10
a[8](); // 10
a[9](); // 10
複製程式碼

所以,你現在還有什麼理由不用let?

有時候const比let更好

const和let的唯一區別就是,const不可以被更改,所以當宣告變數的時候,尤其是在宣告容易被更改的全域性變數的時候,儘量使用const。

  • 更好的程式碼語義化,一眼看到就是常量。
  • 另一個原因是因為JavaScript 編譯器對const的優化要比let好,多使用const,有利於提高程式的執行效率。
  • 所有的函式都應該設定為常量。

動態字串

不要使用“雙引號”,一律用單引號或反引號

// low
const a = "foobar";
const b = 'foo' + a + 'bar';

// best
const a = 'foobar';
const b = `foo${a}bar`;
const c = 'foobar';
複製程式碼

解構賦值的騷操作

變數賦值

在用到陣列成員對變數賦值時,儘量使用解構賦值。

const arr = [1, 2, 3, 4];

// low
const first = arr[0];
const second = arr[1];

// good
const [first, second] = arr;
複製程式碼

函式傳物件

函式的引數如果是物件的成員,優先使用解構賦值。

// low
function getFullName(user) {
  const firstName = user.firstName;
  const lastName = user.lastName;
}

// good
function getFullName({ firstName, lastName }) {
}
複製程式碼

如果函式返回多個值,優先使用物件的解構賦值,而不是陣列的解構賦值。這樣便於以後新增返回值,以及更改返回值的順序。

// low
function processInput(input) {
  return [left, right, top, bottom];
}

// good
function processInput(input) {
  return { left, right, top, bottom };
}

const { left, right } = processInput(input);
複製程式碼

關於物件的細節

逗號

單行定義的物件結尾不要逗號:

// low
const a = { k1: v1, k2: v2, };

// good
const a = { k1: v1, k2: v2 };
複製程式碼

多行定義的物件要保留逗號:

// low
const b = {
  k1: v1,
  k2: v2
};

// good
const b = {
  k1: v1,
  k2: v2,
};
複製程式碼

一次性初始化完全

不要宣告之後又給物件新增新屬性:

// low
const a = {};
a.x = 3;

// good
const a = { x: null };
a.x = 3;
複製程式碼

如果一定非要加請使用Object.assign:

const a = {};
Object.assign(a, { x: 3 });
複製程式碼

如果物件的屬性名是動態的,可以在創造物件的時候,使用屬性表示式定義:

// low
const obj = {
  id: 5,
  name: 'San Francisco',
};
obj[getKey('enabled')] = true;

// good
const obj = {
  id: 5,
  name: 'San Francisco',
  [getKey('enabled')]: true,
};
複製程式碼

再簡潔一點

在定義物件時,能簡潔表達儘量簡潔表達:

var ref = 'some value';

// low
const atom = {
  ref: ref,

  value: 1,

  addValue: function (value) {
    return atom.value + value;
  },
};

// good
const atom = {
  ref,

  value: 1,

  addValue(value) {
    return atom.value + value;
  },
};
複製程式碼

陣列

...

使用擴充套件運算子(...)拷貝陣列:

// 還在用for i 你就太low了
const len = items.length;
const itemsCopy = [];
let i;

for (i = 0; i < len; i++) {
  itemsCopy[i] = items[i];
}

// cool !
const itemsCopy = [...items];
複製程式碼

不要跟我提類陣列

用 Array.from 方法,將類似陣列的物件轉為陣列:

const foo = document.querySelectorAll('.foo');
const nodes = Array.from(foo);
複製程式碼

函式

箭頭函式=>

立即執行函式可以寫成箭頭函式的形式:

(() => {
  console.log('Welcome to the Internet.');
})();
複製程式碼

儘量寫箭頭函式使你的程式碼看起來簡潔優雅:

// low
[1, 2, 3].map(function (x) {
  return x * x;
});

// cool !
[1, 2, 3].map(x => x * x);
複製程式碼

不要把布林值直接傳入函式

// low
function divide(a, b, option = false ) {
}

// good
function divide(a, b, { option = false } = {}) {
}
複製程式碼

別再用arguments(類陣列)了!

使用 rest 運算子(...)代替,rest 運算子可以提供一個真正的陣列。

// low
function concatenateAll() {
  const args = Array.prototype.slice.call(arguments);
  return args.join('');
}

// good
function concatenateAll(...args) {
  return args.join('');
}
複製程式碼

傳參時試試設定預設值?

// low
function handleThings(opts) {
  opts = opts || {};
}

// good
function handleThings(opts = {}) {
  // ...
}
複製程式碼

Object?Map!

簡單的鍵值對優先Map

如果只是簡單的key: value結構,建議優先使用Map,因為Map提供方便的遍歷機制。

let map = new Map(arr);
// 遍歷key值
for (let key of map.keys()) {
  console.log(key);
}
// 遍歷value值
for (let value of map.values()) {
  console.log(value);
}
// 遍歷key和value值
for (let item of map.entries()) {
  console.log(item[0], item[1]);
}
複製程式碼

更加簡潔直觀class語法

// low
function Queue(contents = []) {
  this._queue = [...contents];
}
Queue.prototype.pop = function() {
  const value = this._queue[0];
  this._queue.splice(0, 1);
  return value;
}

// good
class Queue {
  constructor(contents = []) {
    this._queue = [...contents];
  }
  pop() {
    const value = this._queue[0];
    this._queue.splice(0, 1);
    return value;
  }
}
複製程式碼

模組化

引入模組

使用import取代require,因為Module是Javascript模組的標準寫法。

// bad
const moduleA = require('moduleA');
const func1 = moduleA.func1;
const func2 = moduleA.func2;

// good
import { func1, func2 } from 'moduleA';
複製程式碼

輸出模組

使用export輸出變數,拒絕module.exports:

import React from 'react';

class Breadcrumbs extends React.Component {
  render() {
    return <nav />;
  }
};

export default Breadcrumbs;
複製程式碼
  • 輸出單個值,使用export default
  • 輸出多個值,使用export
  • export default與普通的export不要同時使用

編碼規範

  • 模組輸出一個函式,首字母應該小寫:
function getData() {
}

export default getData;
複製程式碼
  • 模組輸出一個物件,首字母應該大寫
const Person = {
  someCode: {
  }
};

export default Person ;
複製程式碼

相關文章