天天看點

JS中的箭頭函數

作者:是蜃樓啊

在JavaScript中,箭頭函數是一種簡化的函數文法,它在ES6(ECMAScript 2015)引入。箭頭函數的文法比傳統的function表達式更簡潔,同時還有一些特性,例如繼承外部作用域的this值。

箭頭函數的基本文法如下:

(param1, param2, ..., paramN) => { statements }
(param1, param2, ..., paramN) => expression           

當箭頭函數隻有一個參數時,可以省略括号:

param => { statements }
param => expression           

當函數體隻包含一個傳回值表達式時,可以省略花括号并直接傳回該表達式的值:

(param1, param2) => param1 + param2           

需要注意的是,箭頭函數有以下特性:

  • 沒有自己的this值:箭頭函數内的this值繼承自包含它的函數作用域,這有助于解決一些this指向問題。
  • 沒有arguments對象:在箭頭函數内部,無法通路傳統函數中的arguments對象。但你可以使用剩餘參數(...rest)文法來擷取參數清單。
  • 不能用作構造函數:箭頭函數不能用作構造函數,是以不能使用new運算符執行個體化。
  • 沒有原型:箭頭函數沒有prototype屬性,因為它們不能作為構造函數使用。

更多箭頭函數的用法

  1. 鍊式調用:

箭頭函數的簡潔文法使得在鍊式調用中使用它們變得更加容易。例如,我們可以在數組上使用多個數組方法并将箭頭函數作為回調函數:

const numbers = [1, 2, 3, 4, 5];

const doubledAndFiltered = numbers
  .map(num => num * 2)
  .filter(num => num > 5);

console.log(doubledAndFiltered); // [6, 8, 10]           
  1. 事件處理程式:

箭頭函數可以友善地用作事件處理程式,因為它們繼承了外部作用域的this值。這樣就避免了使用bind來綁定事件處理程式的需要。例如:

class Button {
  constructor() {
    this.buttonElement = document.createElement('button');
    this.buttonElement.textContent = 'Click me!';
    this.buttonElement.addEventListener('click', () => this.handleClick());
    document.body.appendChild(this.buttonElement);
  }

  handleClick() {
    console.log('Button clicked!');
  }
}

const button = new Button();           
  1. 在數組方法中使用:

常見的數組方法,如filter、reduce、forEach等,也可以與箭頭函數一起使用,以簡化代碼并使其更具可讀性:

const numbers = [1, 2, 3, 4, 5];

const evenNumbers = numbers.filter(num => num % 2 === 0);

console.log(evenNumbers); // [2, 4]

const sum = numbers.reduce((acc, num) => acc + num, 0);

console.log(sum); // 15

numbers.forEach((num, index) => console.log(`Index ${index}: ${num}`));           
  1. 立即調用的箭頭函數:

箭頭函數還可以作為立即調用函數表達式(IIFE)使用,這在某些場景下有助于限制變量的作用域:

const result = (() => {
  const localVar = 'I am only available within this IIFE';
  return localVar.toUpperCase();
})();

console.log(result); // 'I AM ONLY AVAILABLE WITHIN THIS IIFE'           
  1. 對象字面量和箭頭函數:

在箭頭函數中直接傳回對象字面量時,需要注意文法。由于大括号 {} 在箭頭函數中被解釋為代碼塊,而不是對象字面量,是以需要在對象字面量周圍添加額外的括号:

const getObject = () => ({ key: 'value' });

console.log(getObject()); // { key: 'value' }           
  1. 多行箭頭函數:

雖然箭頭函數通常用于簡潔的單行函數,但它們也可以用于多行函數。在這種情況下,需要使用大括号包裹函數體,并在需要傳回值時使用 return 關鍵字:

const addWithLogging = (a, b) => {
  console.log(`Adding ${a} and ${b}`);
  return a + b;
};

console.log(addWithLogging(3, 4)); // 輸出 "Adding 3 and 4",然後輸出 7           
  1. 箭頭函數與解構參數:

箭頭函數可以與解構參數一起使用,可以更簡潔地處理對象或數組。以下是一些示例:

// 對象解構
const users = [
  { id: 1, name: 'Alice', age: 30 },
  { id: 2, name: 'Bob', age: 25 },
  { id: 3, name: 'Charlie', age: 35 },
];

const getUserNames = users.map(({ name }) => name);
console.log(getUserNames); // 輸出:['Alice', 'Bob', 'Charlie']

// 數組解構
const points = [
  [1, 2],
  [3, 4],
  [5, 6],
];

const getDistancesFromOrigin = points.map(([x, y]) => Math.sqrt(x * x + y * y));
console.log(getDistancesFromOrigin); // 輸出:[2.23606797749979, 5, 7.810249675906654]           
  1. 箭頭函數和 this:

由于箭頭函數在其詞法作用域内捕獲 this 值,是以在某些情況下可能導緻問題。例如,在對象方法中使用箭頭函數時,它不會擷取到對象的 this,而是捕獲到外部作用域的 this。為了解決這個問題,需要使用傳統的 function 聲明或表達式。

const obj = {
  value: 10,
  getValue: function() {
    // 正常的function表達式,`this` 指向obj
    return this.value;
  },
  getValueWithArrow: () => {
    // 箭頭函數,`this` 指向外部作用域(在這種情況下是全局對象或undefined)
    return this.value;
  },
};

console.log(obj.getValue()); // 輸出:10
console.log(obj.getValueWithArrow()); // 輸出:undefined(嚴格模式)或全局對象的value屬性           
  1. 箭頭函數作為高階函數的參數:

在處理高階函數時,箭頭函數非常有用,因為它們可以使代碼更簡潔。高階函數是接受一個或多個函數作為參數、傳回一個函數的函數。這裡有一個使用箭頭函數的高階函數示例:

const add = a => b => a + b;

const add5 = add(5);
console.log(add5(3)); // 輸出:8           

在上面的示例中,add 函數接受一個參數 a 并傳回一個新的函數,該函數接受另一個參數 b 并傳回 a + b 的結果。

  1. 不要在所有場景中都使用箭頭函數:

盡管箭頭函數有很多優點,但并非所有場景都适用。以下是一些避免使用箭頭函數的情況:

  • 在需要動态上下文的函數(如事件處理程式)中,箭頭函數繼承了它們的詞法作用域。在這種情況下,可能需要使用 function 聲明或表達式,以便根據需要通路目前上下文。
  • 當需要使用 arguments 對象時,箭頭函數不會建立它。在這種情況下,需要使用傳統的 function 聲明或表達式。

總之,箭頭函數的簡潔文法和特性使得它們在許多情況下都非常有用,還可以用在setTimeout、錯誤處理、Promise中等等。但是,在遇到 this、arguments 或其他相關問題時,有時可能需要使用傳統的function聲明或表達式來解決特定問題。

參考:JS中的箭頭函數 - 掘金

繼續閱讀