宣告變數的新姿勢
用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 ;
複製程式碼