深入理解React中何时使用箭头函数


Posted in Javascript onAugust 23, 2017

前言

相信大家当想起箭头函数时,脑海里可能会浮现 棒,酷,简洁,有趣 等形容词,其实,我们存在一些 更充分的理由 使我们在联想起 箭头函数 时不得不想到的,本文详细的给大家介绍了关于React何时使用箭头函数的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍:

解决 this 引起的问题

箭头函数不会在函数体内重新定义 this 的值,这使得在回调中的行为更容易预测,并且避免了 this 在回调中潜存的 bug

下面我们来看一个 example

我们期望点击按钮,改变按钮颜色,代码如下

class BrokenButton extends React.Component {
 render() {
 return (
  <button onClick={this.handleClick} style={this.state}>
  Set background to red
  </button>
 );
 }

 handleClick() {
 this.setState({ backgroundColor: "red" });
 }
}

render(<BrokenButton />, document.getElementById("root"));

然而,当我们点击按钮时,什么效果都没有,为什么会这样呢

其实,不是 handleClick 方法没有起作用,因为 JavaScript 中压根没有方法, JavaScript 中只有函数,而函数中的 this 存在一些规则,正是这些规则,让上面的 handleClick 中的 this 值变成了 null

你需要清楚明白的是: 你无法确定一个方法函数中 this 的指向,因为它的值跟函数的调用方式有关

除非,你使用 箭头函数,因为箭头函数中 this 的值是继承自 外围作用域

class Button extends React.Component {
 render() {
 return (
  <button
  onClick={() => this.setState({ backgroundColor: "red" })}
  style={this.state}
  >
  Set background to red
  </button>
 );
 }
}

render(<Button />, document.getElementById("root"));

现在就对了,接下来,我们继续

浏览器支持

浏览器对 箭头函数 的支持大概是 73%,因为目前,IE 并不支持。但如果你已经意识到这一点,并且你还会代码转译,这对你来说就不算什么问题

性能问题

大家都发现了,箭头函数 书写起来是非常容易的,但书写忒多的函数,也会造成一些问题

定义函数是昂贵的

浏览器每执行一次 =>,就需要创建一个 新的函数对象,这其实是一个比较 昂贵 的操作

当然,如果你不是想构建一个 性能超级无敌宇宙螺旋棒 的组件,渲染一个 非常长 的列表或 非常大 的表格,你也不会发现这是一个 问题

所以,如果你的组件只是在页面中渲染个几次,你也 没必要忒担心 性能这方面的问题

两个相同的箭头函数并不相等

为了让大家意识到这个问题,接下来,我们用 == 比较一下两个相同的箭头函数相不相等

const a = x => x,
  b = x => x;

render(
 <div>
 <h3>
  Are <code>a</code> and <code>b</code> equal by <code>==</code>?
 </h3>
 <p>
  {a == b ? "Yes!" : "No :("}
 </p>
 </div>,
 document.getElementById("root")
);

如果你在 render 中使用箭头函数,那么你在每次调用 render 时都会去创建一个新的函数对象,此时,即使使用 PureComponent 和 shouldComponentUpdate 也起不到优化作用

你可以在下面实例中看清这一点,其中, <PropChangeCounter /> 组件用于打印 props 改变的次数

import PropChangeCounter from "react-armory-prop-change-counter";

class App extends React.Component {
 constructor(props) {
 super(props);
 this.state = { email: "" };
 }
 render() {
 return (
  <div>
  <input
   placeholder="Email"
   value={this.state.email}
   onChange={e => this.setState({ email: e.target.value })}
  />
  <PropChangeCounter
   constant={"this doesn't change"}
   value={this.state.email}
   onChange={e => this.setState({ email: e.target.value })}
  />
  </div>
 );
 }
}

render(<App />, document.getElementById("root"));

只定义一次

如果你觉得 性能 对你的组件很重要,那么你肯定会想如果在组件中只定义箭头函数 一次 该有多好

其中一种实现方式是在 constructor 中使用箭头函数,当然,对于复杂些的组价来说,这会变的很笨拙

如果你使用了 Babel 或 create-react-app 构建你的应用,你可以将箭头函数设置为 class fields 或 arrow function methods

如下,你可以将 handleClick 重新定义为一个 arrow function method,来修复第一个 example 中的 bug

class Button extends React.Component {
 render() {
 return (
  <button onClick={this.handleClick} style={this.state}>
  Set background to red
  </button>
 );
 }

 // Note: this syntax is not yet part of JavaScript proper, but is slated
 // for inclusion in the next version. It should already work with Babel.
 handleClick = () => {
 this.setState({ backgroundColor: "red" });
 };
}

总结

  • 如果 环境支持 箭头函数,那么鼓励使用
  • 尽量避免对 React 组件 使用箭头函数,它会使 调试 变的困难
  • 如果有需要,可以在 render 中使用箭头函数
  • 为 性能 着想,避免在 render 中使用大量函数

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

原文链接:When should I use Arrow Functions? (James K Nelson)

Javascript 相关文章推荐
Array.prototype.slice.apply的使用方法
Mar 17 Javascript
javascript随机将第一个dom中的图片添加到第二个div中示例
Oct 08 Javascript
jQuery常用操作方法及常用函数总结
Jun 19 Javascript
实例分析javascript中的call()和apply()方法
Nov 28 Javascript
谈谈第三方App接入微信登录 解读
Dec 27 Javascript
node.js基础知识小结
Feb 26 Javascript
node结合swig渲染摸板的方法
Apr 11 Javascript
JS中创建自定义类型的常用模式总结【工厂模式,构造函数模式,原型模式,动态原型模式等】
Jan 19 Javascript
vue router导航守卫(router.beforeEach())的使用详解
Apr 19 Javascript
封装微信小程序http拦截器过程解析
Aug 13 Javascript
vue 路由子组件created和mounted不起作用的解决方法
Nov 05 Javascript
javascript设计模式 ? 代理模式原理与用法实例分析
Apr 16 Javascript
自定义类似于jQuery UI Selectable 的Vue指令v-selectable
Aug 23 #jQuery
JS数组交集、并集、差集的示例代码
Aug 23 #Javascript
关于Vue实现组件信息的缓存问题
Aug 23 #Javascript
详解webpack进阶之loader篇
Aug 23 #Javascript
Vue中定义全局变量与常量的各种方式详解
Aug 23 #Javascript
基于JavaScript实现带数据验证和复选框的表单提交
Aug 23 #Javascript
使用JS组件实现带ToolTip验证框的实例代码
Aug 23 #Javascript
You might like
php调用dll的实例操作动画与代码分享
2012/08/14 PHP
Zend Studio 实用快捷键一览表(精心整理)
2013/08/10 PHP
DWZ+ThinkPHP开发时遇到的问题分析
2016/12/12 PHP
PhpStorm+xdebug+postman调试技巧分享
2020/09/15 PHP
php解析非标准json、非规范json的方式实例
2020/12/10 PHP
兼容IE和Firefox火狐的上下、左右循环无间断滚动JS代码
2013/04/19 Javascript
jQuery图片滚动图片的效果(另类实现)
2013/06/02 Javascript
js购物车实现思路及代码(个人感觉不错)
2013/12/23 Javascript
Javascript基础知识(三)BOM,DOM总结
2014/09/29 Javascript
javascript+ajax实现产品页面加载信息
2015/07/09 Javascript
微信端开发--登录小程序步骤
2017/01/11 Javascript
Vue的事件响应式进度条组件实例详解
2018/02/04 Javascript
详解关于element级联选择器数据回显问题
2019/02/20 Javascript
VUE实现移动端列表筛选功能
2019/08/23 Javascript
Vue-CLI与Vuex使用方法实例分析
2020/01/06 Javascript
Node.js 在本地生成日志文件的方法
2020/02/07 Javascript
[36:29]2018DOTA2亚洲邀请赛 4.1 小组赛 A组加赛 LGD vs TNC
2018/04/02 DOTA
python三元运算符实现方法
2013/12/17 Python
Python内置的字符串处理函数详细整理(覆盖日常所用)
2014/08/19 Python
python测试驱动开发实例
2014/10/08 Python
使用Python编写一个在Linux下实现截图分享的脚本的教程
2015/04/24 Python
Python字符串格式化
2015/06/15 Python
python开发之文件操作用法实例
2015/11/13 Python
决策树剪枝算法的python实现方法详解
2019/09/18 Python
python GUI库图形界面开发之PyQt5菜单栏控件QMenuBar的详细使用方法与实例
2020/02/28 Python
SpringBoot实现登录注册常见问题解决方案
2020/03/04 Python
Python Django搭建网站流程图解
2020/06/13 Python
python用tkinter实现一个gui的翻译工具
2020/10/26 Python
matplotlib交互式数据光标实现(mplcursors)
2021/01/13 Python
css3背景_动力节点Java学院整理
2017/07/11 HTML / CSS
英国标准协会商店:BSI Shop
2019/02/25 全球购物
旺仔牛奶广告词
2014/03/20 职场文书
先进班集体事迹材料
2014/12/25 职场文书
安全教育主题班会教案
2015/08/12 职场文书
Python学习之包与模块详解
2022/03/19 Python
总结三种用 Python 作为小程序后端的方式
2022/05/02 Python