> 웹 프론트엔드 > 프런트엔드 Q&A > 화살표 기능은 es6에 속합니까?

화살표 기능은 es6에 속합니까?

青灯夜游
풀어 주다: 2023-01-18 19:53:49
원래의
917명이 탐색했습니다.

화살표 기능은 es6에 속합니다. 화살표 함수는 ES6에 도입된 새로운 기능입니다. "=>" 화살표를 사용하여 함수를 정의하세요. 예를 들어 "var f = v => v;"는 "var f = function (v) { return v;}; "; 화살표 함수에 매개변수가 필요하지 않거나 여러 매개변수가 필요한 경우 괄호를 사용하여 매개변수 부분을 나타냅니다(예: "var f = () =>).

화살표 기능은 es6에 속합니까?

이 튜토리얼의 운영 환경: Windows 7 시스템, ECMAScript 버전 6, Dell G3 컴퓨터.

화살표 함수

화살표 함수는 ES6에 도입된 새로운 기능으로 "화살표"(=>)를 사용하여 함수를 정의합니다. 간결한 구문과 this 키워드 처리로 인해 화살표 함수는 개발자들 사이에서 빠르게 선호되는 기능이 되었습니다.

var f = v => v;
// 等同于
var f = function (v) {
  return v;
};
로그인 후 복사

화살표 함수에 매개변수가 필요하지 않거나 여러 매개변수가 필요한 경우 괄호를 사용하여 매개변수 부분을 표현하세요.

var f = () => 5;
// 等同于
var f = function () { return 5 };
var sum = (num1, num2) => num1 + num2;
// 等同于
var sum = function(num1, num2) {
  return num1 + num2;
};
로그인 후 복사

화살표 함수의 코드 블록에 둘 이상의 명령문이 있는 경우 중괄호를 사용하여 묶고 return 문을 사용하여 반환하세요.

var sum = (num1, num2) => { return num1 + num2; }
로그인 후 복사

중괄호는 코드 블록으로 해석되므로 화살표 함수가 직접 객체를 반환하는 경우 객체 외부에 괄호를 추가해야 하며, 그렇지 않으면 오류가 보고됩니다.

// 报错
let getTempItem = id => { id: id, name: "Temp" };
// 不报错
let getTempItem = id => ({ id: id, name: "Temp" });
로그인 후 복사

다음은 특별한 경우입니다. 작동하더라도 잘못된 결과를 얻게 됩니다.

let foo = () => { a: 1 };
foo() // undefined
로그인 후 복사

위 코드에서 원래 의도는 객체 { a: 1 } 을 반환하는 것이지만 엔진은 중괄호가 코드 블록이라고 생각하기 때문에 명령문 a: 1 한 줄이 실행됩니다. 이때 a는 문의 레이블로 해석될 수 있으므로 실제로 실행되는 문은 1;이 되며, 그 이후에는 반환값 없이 함수가 종료된다.

화살표 함수에 명령문이 한 줄만 있고 값을 반환할 필요가 없는 경우 중괄호를 쓰지 않고 다음 작성 방법을 사용할 수 있습니다.

let fn = () => void doesNotReturn();
로그인 후 복사

설명:

화살표 함수는 함수형 프로그래밍의 표현입니다. 함수형 프로그래밍은 프로세스의 일부 요소를 생략하고 입력과 출력 간의 관계에 더 중점을 두므로 화살표 함수에는 자체 this가 없습니다. , 인수, 새 대상(ES6) 및 슈퍼(ES6). 화살표 함수는 익명 함수와 동일하므로 new를 생성자로 사용할 수 없습니다.

화살표 함수의 This는 항상 상위 범위에서 this를 가리킵니다. 즉, 화살표 함수는 자신이 위치한 컨텍스트의 this 값을 자신의 this 값으로 캡처합니다. 호출(), 바인드(), 적용()과 같은 어떤 메서드로도 해당 포인터를 변경할 수 없습니다. 화살표 함수에서 이것을 호출하면 단순히 범위 체인을 검색하여 가장 가까운 this를 찾아 사용합니다. 호출 컨텍스트와는 아무 관련이 없습니다. 코드로 설명해보자.

암시적 반환

함수 본문에 표현식이 하나만 있는 경우 ES6 화살표 구문을 더 간결하게 만들 수 있습니다. 모든 내용을 한 줄에 입력하고 중괄호를 제거하고 return 키워드를 제거할 수 있습니다. return关键字。

你已经在上面的示例中看到了这些漂亮的一行代码是如何工作的。下面的orderByLikes()函数返回奈飞剧集对象的数组,按照最高点赞数排序:

// using the JS sort() function to sort the titles in descending order 
// according to the number of likes (more likes at the top, fewer at the bottom
const orderByLikes = netflixSeries.sort((a, b) => b.likes - a.likes)

// call the function 
// output:the titles and the n. of likes in descending order
console.log(orderByLikes)
로그인 후 복사

这种写法很酷,但是要注意代码的可读性。特别是在使用单行和无括号的ES6箭头语法对一堆箭头函数进行排序时。就像这个例子:

const greeter = greeting => name => `${greeting}, ${name}!`
로그인 후 복사

那里发生了什么?尝试使用常规的函数语法:

function greeter(greeting) {
  return function(name) {
    return `${greeting}, ${name}!` 
  }
}
로그인 후 복사

现在,你可以快速看到外部函数greeter如何具有参数greeting,并返回一个匿名函数。这个内部函数又有一个叫做name的参数,并使用greetingname的值返回一个字符串。下面是调用函数的方式:

const myGreet = greeter('Good morning')
console.log( myGreet('Mary') )   

// output: 
"Good morning, Mary!"
로그인 후 복사

注意隐式返回错误

当你的JavaScript箭头函数包含不止一个语句,你需要在大括号内包裹所有语句,并使用return关键字。

在下面的代码中,该函数建立了一个包含几个Netflix剧集的标题和摘要的对象:

const seriesList = netflixSeries.map( series => {
  const container = {}
  container.title = series.name 
  container.summary = series.summary

  // explicit return
  return container
} )
로그인 후 복사

.map()函数中的箭头函数在一系列的语句中展开,在语句的最后返回一个对象。这使得在函数主体周围使用大括号是不可避免的。

另外,由于正在使用花括号,隐式返回便不是一个选项。你必须显式使用return关键字。

如果你的函数使用隐式返回来返回一个对象字面量,你需要使用圆括号来包裹该对象字面量。不这样做将导致错误,因为JavaScript引擎将对象字面量的大括号错误地解析为函数的大括号。正如你刚才注意到的,当你在一个箭头函数中使用大括号时,你不能省略return关键字。

前面代码的较短版本演示了这种语法:

// Uncaught SyntaxError: unexpected token: ':'
const seriesList = netflixSeries.map(series => { title: series.name });

// Works fine
const seriesList = netflixSeries.map(series => ({ title: series.name }));
로그인 후 복사

无法命名箭头函数

function

위의 예에서 이러한 멋진 한 줄짜리 코드가 어떻게 작동하는지 확인했습니다. 다음 orderByLikes() 함수는 좋아요 수가 가장 많은 순서로 정렬된 Netflix 에피소드 개체 배열을 반환합니다.

const anonymous = function() {
  return 'You can\'t identify me!' 
}
로그인 후 복사

이 작성 방법은 멋지지만 코드의 가독성에 주의하세요. 특히 한 줄과 대괄호 없는 ES6 화살표 구문을 사용하여 여러 화살표 함수를 정렬할 때 더욱 그렇습니다. 예를 들면 다음과 같습니다. 🎜
const anonymousArrowFunc = () => 'You can\'t identify me!'
로그인 후 복사
로그인 후 복사
🎜거기서 무슨 일이 일어났나요? 일반 함수 구문을 사용해 보세요. 🎜
console.log(anonymousArrowFunc.name)
// output: "anonymousArrowFunc"
로그인 후 복사
로그인 후 복사
🎜이제 외부 함수 greetergreeting 매개변수가 있고 익명 함수를 반환하는 방법을 빠르게 확인할 수 있습니다. 이 내부 함수는 name이라는 또 다른 매개변수를 취하고 greetingname 값을 사용하여 문자열을 반환합니다. 함수를 호출하는 방법은 다음과 같습니다. 🎜
let counter = 5
let countDown = setInterval(() => {
  console.log(counter)
  counter--
  if (counter === 0) {
    console.log("I have no name!!")
    clearInterval(countDown)
  }
}, 1000)
로그인 후 복사
로그인 후 복사

🎜암시적 반환 오류에 유의하세요🎜

🎜JavaScript 화살표 함수에 둘 이상의 문이 포함된 경우 이를 래핑해야 합니다. 중괄호 모든 명령문을 return 키워드로 묶습니다. 🎜🎜아래 코드에서 이 함수는 여러 Netflix 에피소드의 제목과 요약을 포함하는 객체를 생성합니다. 🎜
<button class="start-btn">Start Counter</button>

...

const startBtn = document.querySelector(".start-btn");

startBtn.addEventListener(&#39;click&#39;, function() {
  this.classList.add(&#39;counting&#39;)
  let counter = 5;
  const timer = setInterval(() => {
    this.textContent = counter 
    counter -- 
    if(counter < 0) {
      this.textContent = &#39;THE END!&#39;
      this.classList.remove(&#39;counting&#39;)
      clearInterval(timer)
    }
  }, 1000) 
})
로그인 후 복사
로그인 후 복사
🎜 .map() 함수의 화살표 함수는 일련의 명령문에서 사용됩니다. 명령문 끝에 객체를 반환합니다. 이로 인해 함수 본문 주위에 중괄호를 사용하는 것이 불가피해졌습니다. 🎜🎜또한 중괄호를 사용하고 있으므로 암시적 반환은 옵션이 아닙니다. return 키워드를 명시적으로 사용해야 합니다. 🎜🎜함수가 암시적 반환을 사용하여 객체 리터럴을 반환하는 경우 괄호를 사용하여 객체 리터럴을 래핑해야 합니다. 그렇게 하지 않으면 JavaScript 엔진이 객체 리터럴의 중괄호를 함수의 중괄호로 잘못 구문 분석하기 때문에 오류가 발생합니다. 방금 알아차린 것처럼 화살표 함수에 중괄호를 사용할 때는 return 키워드를 생략할 수 없습니다. 🎜🎜이전 코드의 더 짧은 버전은 다음 구문을 보여줍니다. 🎜
startBtn.addEventListener(&#39;click&#39;, function() {
  console.log(this)
  ...
})
로그인 후 복사
로그인 후 복사

🎜화살표 함수의 이름을 지정할 수 없습니다. 🎜🎜🎜function 키워드 및 인수 목록에서 이름으로 식별되지 않는 함수를 익명 함수라고 합니다. 일반 익명 함수 표현식은 다음과 같습니다. 🎜
startBtn.addEventListener(&#39;click&#39;, () => {
  console.log(this)
  ...
})
로그인 후 복사
로그인 후 복사
🎜화살표 함수는 모두 익명 함수입니다. 🎜
const anonymousArrowFunc = () => &#39;You can\&#39;t identify me!&#39;
로그인 후 복사
로그인 후 복사

从ES6开始,变量和方法可以通过匿名函数的语法位置,使用name属性来推断其名称。这使得在检查函数值或报告错误时有可能识别该函数。

使用anonymousArrowFunc检查一下:

console.log(anonymousArrowFunc.name)
// output: "anonymousArrowFunc"
로그인 후 복사
로그인 후 복사

需要注意的是,只有当匿名函数被分配给一个变量时,这个可以推断的name属性才会存在,正如上面的例子。如果你使用匿名函数作为回调函数,你就会失去这个有用的功能。在下面的演示中,.setInterval()方法中的匿名函数无法利用name属性:

let counter = 5
let countDown = setInterval(() => {
  console.log(counter)
  counter--
  if (counter === 0) {
    console.log("I have no name!!")
    clearInterval(countDown)
  }
}, 1000)
로그인 후 복사
로그인 후 복사

这还不是全部。这个推断的name属性仍然不能作为一个适当的标识符,你可以用它来指代函数本身--比如递归、解除绑定事件等。

如何处理this关键字

关于箭头函数,最重要的一点是它们处理this关键字的方式。特别是,箭头函数内的this关键字不会重新绑定。

为了说明这意味着什么,请查看下面的演示。

这里有一个按钮。点击按钮会触发一个从5到1的反向计数器,它显示在按钮本身。

<button class="start-btn">Start Counter</button>

...

const startBtn = document.querySelector(".start-btn");

startBtn.addEventListener(&#39;click&#39;, function() {
  this.classList.add(&#39;counting&#39;)
  let counter = 5;
  const timer = setInterval(() => {
    this.textContent = counter 
    counter -- 
    if(counter < 0) {
      this.textContent = &#39;THE END!&#39;
      this.classList.remove(&#39;counting&#39;)
      clearInterval(timer)
    }
  }, 1000) 
})
로그인 후 복사
로그인 후 복사

注意到.addEventListener()方法里面的事件处理器是一个常规的匿名函数表达式,而不是一个箭头函数。为什么呢?如果在函数内部打印this的值,你会看到它引用了监听器所连接的按钮元素,这正是我们所期望的,也是程序按计划工作所需要的:

startBtn.addEventListener(&#39;click&#39;, function() {
  console.log(this)
  ...
})
로그인 후 복사
로그인 후 복사

下面是它在Firefox开发人员工具控制台中的样子:

화살표 기능은 es6에 속합니까?

然后,尝试使用箭头函数来替代常规函数,就像这样:

startBtn.addEventListener(&#39;click&#39;, () => {
  console.log(this)
  ...
})
로그인 후 복사
로그인 후 복사

现在,this不再引用按钮元素。相反,它引用Window对象:

화살표 기능은 es6에 속합니까?

这意味着,如果你想要在按钮被点击之后,使用this来为按钮添加class,你的代码就无法正常工作:

// change button&#39;s border&#39;s appearance
this.classList.add(&#39;counting&#39;)
로그인 후 복사

下面是控制台中的错误信息:

화살표 기능은 es6에 속합니까?

当你在JavaScript中使用箭头函数,this关键字的值不会被重新绑定。它继承自父作用域(也称为词法作用域)。在这种特殊情况下,箭头函数被作为参数传递给startBtn.addEventListener()方法,该方法位于全局作用域中。因此,函数处理器中的this也被绑定到全局作用域中--也就是Window对象。

因此,如果你想让this引用程序中的开始按钮,正确的做法是使用一个常规函数,而不是一个箭头函数。

匿名箭头函数

在上面的演示中,接下来要注意的是.setInterval()方法中的代码。在这里,你也会发现一个匿名函数,但这次是一个箭头函数。为什么?

请注意,如果你使用常规函数,this值会是多少:

const timer = setInterval(function() {
  console.log(this)
  ...
}, 1000)
로그인 후 복사

button元素吗?并不是。这个值将会是Window对象!

事实上,上下文已经发生了变化,因为现在this在一个非绑定的或全局的函数中,它被作为参数传递给.setInterval() 。因此,this关键字的值也发生了变化,因为它现在被绑定到全局作用域。

在这种情况下,一个常见的hack手段是包括另一个变量来存储this关键字的值,这样它就会一直指向预期的元素--在这种情况下,就是button元素:

const that = this
const timer = setInterval(function() {
  console.log(that)
  ...
}, 1000)
로그인 후 복사

你也可以使用.bind()来解决这个问题:

const timer = setInterval(function() {
  console.log(this)
  ...
}.bind(this), 1000)
로그인 후 복사

有了箭头函数,问题就彻底消失了。下面是使用箭头函数时this的值:

const timer = setInterval( () => { 
  console.log(this)
  ...
}, 1000)
로그인 후 복사

화살표 기능은 es6에 속합니까?

这次,控制台打印了button,这就是我们想要的。事实上,程序要改变按钮的文本,所以它需要this来指代button元素:

const timer = setInterval( () => { 
  console.log(this)
 // the button&#39;s text displays the timer value
  this.textContent = counter
}, 1000)
로그인 후 복사

箭头函数没有自己的this上下文。它们从父级继承this的值,正是因为这个特点,在上面这种情况下就是很好的选择。

不正常工作的情况

箭头函数并不只是在JavaScript中编写函数的一种花里胡哨的新方法。它们有自己的局限性,这意味着在有些情况下你不想使用箭头函数。让我们看看更多的例子。

箭头函数作为对象方法

箭头函数作为对象上的方法不能很好地工作。

考虑这个netflixSeries对象,上面有一些属性和一系列方法。调用console.log(netflixSeries.getLikes()) 应该会打印一条信息,说明当前喜欢的人数。console.log(netflixSeries.addLike())应该会增加一个喜欢的人数,然后在控制台上打印新值:

const netflixSeries = {
  title: &#39;After Life&#39;, 
  firstRealease: 2019,
  likes: 5,
  getLikes: () => `${this.title} has ${this.likes} likes`,
  addLike: () => {  
    this.likes++
    return `Thank you for liking ${this.title}, which now has ${this.likes} likes`
  } 
}
로그인 후 복사

相反,调用.getLikes()方法返回&#39;undefined has NaN likes&#39;,调用.addLike()方法返回&#39;Thank you for liking undefined, which now has NaN likes&#39;。因此,this.titlethis.likes未能分别引用对象的属性titlelikes

这次,问题出在箭头函数的词法作用域上。对象方法中的this引用的是父对象的范围,在本例中是Window对象,而不是父对象本身--也就是说,不是netflixSeries对象。

当然,解决办法是使用常规函数:

const netflixSeries = {
  title: &#39;After Life&#39;, 
  firstRealease: 2019,
  likes: 5,
  getLikes() {
    return `${this.title} has ${this.likes} likes`
  },
  addLike() { 
    this.likes++
    return `Thank you for liking ${this.title}, which now has ${this.likes} likes`
  } 
}

// call the methods 
console.log(netflixSeries.getLikes())
console.log(netflixSeries.addLike())

// output: 
After Life has 5 likes
Thank you for liking After Life, which now has 6 likes
로그인 후 복사

箭头函数与第三方库

另一个需要注意的问题是,第三方库通常会绑定方法调用,因此this值会指向一些有用的东西。

比如说,在Jquery事件处理器内部,this将使你能够访问处理器所绑定的DOM元素:

$(&#39;body&#39;).on(&#39;click&#39;, function() {
  console.log(this)
})
// <body>
로그인 후 복사

但是如果我们使用箭头函数,正如我们所看到的,它没有自己的this上下文,我们会得到意想不到的结果:

$(&#39;body&#39;).on(&#39;click&#39;, () =>{
  console.log(this)
})
// Window
로그인 후 복사

下面是使用Vue的其他例子:

new Vue({
  el: app,
  data: {
    message: &#39;Hello, World!&#39;
  },
  created: function() {
    console.log(this.message);
  }
})
// Hello, World!
로그인 후 복사

created钩子内部,this被绑定到Vue实例上,因此会显示&#39;Hello, World!&#39;信息。

然而如果我们使用箭头函数,this将会指向父作用域,上面没有message属性:

new Vue({
  el: app,
  data: {
    message: &#39;Hello, World!&#39;
  },
  created: () => {
    console.log(this.message);
  }
})
// undefined
로그인 후 복사

箭头函数没有arguments对象

有时,你可能需要创建一个具有无限参数个数的函数。比如,假设你想创建一个函数,列出你最喜欢的奈飞剧集,并按照偏好排序。然而,你还不知道你要包括多少个剧集。JavaScript提供了arguments对象。这是一个类数组对象(不是完整的数组),在调用时存储传递给函数的值。

尝试使用箭头函数实现此功能:

const listYourFavNetflixSeries = () => {
  // we need to turn the arguments into a real array 
  // so we can use .map()
  const favSeries = Array.from(arguments) 
  return favSeries.map( (series, i) => {
    return `${series} is my #${i +1} favorite Netflix series`  
  } )
  console.log(arguments)
}

console.log(listYourFavNetflixSeries(&#39;Bridgerton&#39;, &#39;Ozark&#39;, &#39;After Life&#39;))
로그인 후 복사

当你调用该函数时,你会得到以下错误:Uncaught ReferenceError: arguments is not defined。这意味着arguments对象在箭头函数中是不可用的。事实上,将箭头函数替换成常规函数就可以了:

const listYourFavNetflixSeries = function() {
   const favSeries = Array.from(arguments) 
   return favSeries.map( (series, i) => {
     return `${series} is my #${i +1} favorite Netflix series`  
   } )
   console.log(arguments)
 }
console.log(listYourFavNetflixSeries(&#39;Bridgerton&#39;, &#39;Ozark&#39;, &#39;After Life&#39;))

// output: 
["Bridgerton is my #1 favorite Netflix series",  "Ozark is my #2 favorite Netflix series",  "After Life is my #3 favorite Netflix series"]
로그인 후 복사

因此,如果你需要arguments对象,你不能使用箭头函数。

但如果你真的想用一个箭头函数来复制同样的功能呢?你可以使用ES6剩余参数(...)。下面是你该如何重写你的函数:

const listYourFavNetflixSeries = (...seriesList) => {
   return seriesList.map( (series, i) => {
     return `${series} is my #${i +1} favorite Netflix series`
   } )
 }
로그인 후 복사

总结

通过使用箭头函数,你可以编写带有隐式返回的单行代码,以解决JavaScript中this关键字的绑定问题。箭头函数在数组方法中也很好用,如.map().sort().forEach().filter()、和.reduce()。但请记住:箭头函数并不能取代常规的JavaScript函数。记住,只有当箭形函数是正确的工具时,才能使用它。

【相关推荐:javascript视频教程编程视频

위 내용은 화살표 기능은 es6에 속합니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

관련 라벨:
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿