深入理解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 相关文章推荐
img的onload的另类用法
Jan 10 Javascript
jQuery生成asp.net服务器控件的代码
Feb 04 Javascript
javascript数字时钟示例分享
Apr 23 Javascript
js树插件zTree获取所有选中节点数据的方法
Jan 28 Javascript
JS+CSS实现简易实用的滑动门菜单效果
Sep 18 Javascript
javascript实现文字无缝滚动
Dec 27 Javascript
BootStrap入门学习第一篇
Aug 28 Javascript
vue2.0在没有dev-server.js下的本地数据配置方法
Feb 23 Javascript
Node.js命令行/批处理中如何更改Linux用户密码浅析
Jul 22 Javascript
JavaScript事件对象event用法分析
Jul 27 Javascript
ES6 class类链式继承,实例化及react super(props)原理详解
Feb 15 Javascript
vue3弹出层V3Popup实例详解
Jan 04 Vue.js
自定义类似于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的FTP学习(二)[转自奥索]
2006/10/09 PHP
如何做到多笔资料的同步
2006/10/09 PHP
PHP无法访问远程mysql的问题分析及解决
2013/05/16 PHP
ThinkPHP3.1的Widget新用法
2014/06/19 PHP
PHP自带函数给数字或字符串自动补齐位数
2014/07/29 PHP
利用 fsockopen() 函数开放端口扫描器的实例
2017/08/19 PHP
去除链接虚线全面分析总结
2006/08/15 Javascript
如何用javascript控制上传文件的大小
2006/10/26 Javascript
获取JavaScript用户自定义类的类名称的代码
2007/03/08 Javascript
jquery解析XML字符串和XML文件的方法说明
2014/02/21 Javascript
jQuery制作仿Mac Lion OS滚动条效果
2015/02/10 Javascript
如何动态加载外部Javascript文件
2015/12/02 Javascript
js实现微信分享代码
2020/10/11 Javascript
javascript数组去重方法分析
2016/12/15 Javascript
js 判断数据类型的几种方法
2017/01/13 Javascript
详解Vue.js入门环境搭建
2017/03/17 Javascript
javaScript canvas实现(画笔大小 颜色 橡皮的实例)
2017/11/28 Javascript
vue左右侧联动滚动的实现代码
2018/06/06 Javascript
angularJs中ng-model-options设置数据同步的方法
2018/09/30 Javascript
彻底弄懂 JavaScript 执行机制
2018/10/23 Javascript
微信小程序车牌号码模拟键盘输入功能的实现代码
2018/11/11 Javascript
在vue-cli3中使用axios获取本地json操作
2020/07/30 Javascript
python基础教程之lambda表达式使用方法
2014/02/12 Python
Python的Tornado框架实现图片上传及图片大小修改功能
2016/06/30 Python
numpy中的高维数组转置实例
2018/04/17 Python
python打开使用的方法
2019/09/30 Python
Tensorflow进行多维矩阵的拆分与拼接实例
2020/02/07 Python
Canvas与图片压缩的示例代码
2017/11/28 HTML / CSS
教育技术职业规划范文
2014/03/04 职场文书
学员自我鉴定
2014/03/19 职场文书
小学师德标兵先进事迹材料
2014/05/25 职场文书
防火标语大全
2014/10/06 职场文书
2014群众路线学习笔记
2014/11/06 职场文书
展览会邀请函
2015/02/02 职场文书
红色故事汇观后感
2015/06/18 职场文书
nginx服务器的下载安装与使用详解
2021/08/02 Servers