摘要:换句话说,箭头函数构造函数调用没有意义,而且是模糊的。让我们看看如果尝试这样做会发生什么执行,其中是一个箭头函数,抛出一个错误,不能用作构造函数。当需要动态上下文时,不能使用箭头函数定义方法,使用构造函数创建对象,在处理事件时从获取目标。
为了保证的可读性,本文采用意译而非直译。
想阅读更多优质文章请猛戳GitHub博客,一年百来篇优质文章等着你!
这些年来,ES6 将 JS 的可用性提升到一个新的水平时: 箭头函数、类等等,这些都很棒。
箭头函数是最有价值的新功能之一,有很多好文章描述了它的上下文透明性和简短的语法。
但每个事务都有两面。通常,新特性会带来一些混乱,其中之一就是箭头函数被误导了。本文将介绍一些场景,在这些场景中,你应该绕过箭头函数,转而使用良好的旧函数表达式或较新的简写语法。并且要注意缩短代码,因为这会影响代码的可读性。
1.定义对象上的方法在JS中,方法是存储在对象属性中的函数。当调用该方法时,this 将指向该方法所属的对象。
Object literal由于箭头函数语法简短,所以使用它来定义方法是很有吸引力的,让咱们来试一试:
const calculate = { array: [1, 2, 3], sum: () => { console.log(this === window); // => true return this.array.reduce((result, item) => result + item); } }; console.log(this === window); // => true // Throws "TypeError: Cannot read property "reduce" of undefined" calculate.sum();
calculate.sum方法用箭头函数定义。 但是在调用时,calculate.sum() 会抛出一个TypeError,因为this.array 为undefined。
当调用calculate对象上的方法sum()时,上下文仍然是 window。之所以会发生这种情况,是因为箭头函数按词法作用域将上下文绑定到 window 对象。
执行this.array等同于window.array,它是undefined。
解决方法是使用常规函数表达式来定义方法。 this 是在调用时确定的,而不是由封闭的上下文决定的,来看看修复后的版本:
const calculate = { array: [1, 2, 3], sum() { console.log(this === calculate); // => true return this.array.reduce((result, item) => result + item); } }; calculate.sum(); // => 6
因为sum是常规函数,所以在调用 calculate.sum() 时 this 是 calculate 对象。 this.array是数组引用,因此正确计算元素之和:6。
Object prototype同样的规则也适用于在原型对象上定义方法。使用一个箭头函数来定义sayCatName方法,this 指向 window
function MyCat(name) { this.catName = name; } MyCat.prototype.sayCatName = () => { console.log(this === window); // => true return this.catName; }; const cat = new MyCat("Mew"); cat.sayCatName(); // => undefined
使用早期的方式定义函数表达式:
function MyCat(name) { this.catName = name; } MyCat.prototype.sayCatName = function() { console.log(this === cat); // => true return this.catName; }; const cat = new MyCat("Mew"); cat.sayCatName(); // => "Mew"
sayCatName常规函数在作为方法调用时将上下文更改为cat对象:cat.sayCatName()。
2. 动态上下文的回调函数this 在JS中是一个强大的特性,它允许根据调用函数的方式更改上下文。通常,上下文是调用发生的目标对象,这使得代码更加自然,就像这个对象发生了什么。
但是,箭头函数会在声明上静态绑定上下文,并且无法使其动态化,但这种方式有坏也有好,有时候我们需要动态绑定。
在客户端编程中,将事件侦听器附加到DOM元素是一项常见的任务。事件触发处理程序函数,并将this作为目标元素,这里如果使用箭头函数就不够灵活。
下面的示例尝试为这样的处理程序使用箭头函数:
const button = document.getElementById("myButton"); button.addEventListener("click", () => { console.log(this === window); // => true this.innerHTML = "Clicked button"; });
在全局上下文中 this 指向 window。 当发生单击事件时,浏览器尝试使用按钮上下文调用处理函数,但箭头函数不会更改其预定义的上下文。this.innerHTML相当于window.innerHTML,没有任何意义。
必须应用函数表达式,该表达式允许根据目标元素更改 this:
const button = document.getElementById("myButton"); button.addEventListener("click", function() { console.log(this === button); // => true this.innerHTML = "Clicked button"; });
当用户单击按钮时,处理程序函数中的 this 指向 button。因此这个问题。innerHTML = "Clicked button" 正确地修改按钮文本以反映已单击状态。
3.调用构造函数this 在构造调用中是新创建的对象。当执行new MyFunction()时,构造函数MyFunction的上下文是一个新对象:this instanceof MyFunction === true。
注意,箭头函数不能用作构造函数。 JavaScript通过抛出异常隐式阻止这样做。
无论如何,this是来自封闭上下文的设置,而不是新创建的对象。换句话说,箭头函数构造函数调用没有意义,而且是模糊的。
让我们看看如果尝试这样做会发生什么:
const Message = (text) => { this.text = text; }; // Throws "TypeError: Message is not a constructor" const helloMessage = new Message("Hello World!");
执行new Message("Hello World!"),其中Message是一个箭头函数,JavaScript抛出一个 TypeError 错误,Message不能用作构造函数。
上面的例子可以使用函数表达式来修复,这是创建构造函数的正确方法(包括函数声明):
const Message = function(text) { this.text = text; }; const helloMessage = new Message("Hello World!");简写语法
箭头函数有一个很好的属性,它可以省略参数圆括号()、块大括号{},如果函数主体只有一条语句,则返回。这有助于编写非常短的函数。
原文作者的大学编程教授给学生一个有趣的任务:编写 用C语言计算字符串长度的最短函数,这是学习和探索新语言的好方式。
然而,在实际应用程序中,许多开发人员都会阅读代码。 最短的语法并不总是适合帮助你的同事即时了解该方法的用途。
在某种程度上,简写的函数变得难以阅读,所以尽量不要过度使用。让各位们看一个例子
const multiply = (a, b) => b === undefined ? b => a * b : a * b; const double = multiply(2); double(3); // => 6 multiply(2, 3); // => 6
multiply返回两个数字的乘法结果或与第一个参数绑定的闭包,以便以后的乘法运算。
该函数运行良好,看起来很短。但从一开始就很难理解它是做什么的。
为了使其更具可读性,可以从箭头函数恢复可选花括号和return语句,或使用常规函数:
function multiply(a, b) { if (b === undefined) { return function(b) { return a * b; } } return a * b; } const double = multiply(2); double(3); // => 6 multiply(2, 3); // => 6
在简短和冗长之间找到一个平衡点是很好的,这样可以使代码更加直观。
总结毫无疑问,箭头函数是一个很好的补充。当正确使用时,它会使前面必须使用.bind()或试图捕获上下文的地方变得简单,它还简化了代码。
某些情况下的优点会给其他情况带来不利。 当需要动态上下文时,不能使用箭头函数:定义方法,使用构造函数创建对象,在处理事件时从 this 获取目标。
代码部署后可能存在的BUG没法实时知道,事后为了解决这些BUG,花了大量的时间进行log 调试,这边顺便给大家推荐一个好用的BUG监控工具 Fundebug。
交流干货系列文章汇总如下,觉得不错点个Star,欢迎 加群 互相学习。
https://github.com/qq44924588...
我是小智,公众号「大迁世界」作者,对前端技术保持学习爱好者。我会经常分享自己所学所看的干货,在进阶的路上,共勉!
关注公众号,后台回复福利,即可看到福利,你懂的。
附:新文章会提前一天发到公众号
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/106377.html
摘要:我们在需要动态上下文的地方不能使用箭头函数定义需要动态上下文的函数,构造函数,需要对象作为目标的回调函数以及用箭头函数难以理解的语句。 从开始接触es6到在项目中使用已经有一段时间了,es6有很多优秀的新特性,其中最有价值的特性之一就是箭头函数,他简洁的语法以及更好理解的this值都非常的吸引我。但是新事物也是有两面性的,箭头函数有他的便捷有他的优点,但是他也有缺点,他的优点是代码简洁...
摘要:当我们想起箭头函数时,脑海里可能会浮现棒,酷,简洁,有趣等形容词,其实,我们存在一些更充分的理由使我们在联想起箭头函数时不得不想到的解决引起的问题箭头函数不会在函数体内重新定义的值,这使得在回调中的行为更容易预测,并且避免了在回调中潜存的下 当我们想起箭头函数时,脑海里可能会浮现 棒,酷,简洁,有趣 等形容词,其实,我们存在一些 更充分的理由 使我们在联想起 箭头函数 时不得不想到的 ...
摘要:显然,箭头函数是不能用来做构造函数,实际上会禁止你这么做,如果你这么做了,它就会抛出异常。换句话说,箭头构造函数的执行并没有任何意义,并且是有歧义的。 showImg(https://segmentfault.com/img/remote/1460000009180813); 共 2670 字,读完需 5 分钟。编译自 Dmitri Pavlutin 的文章,对原文内容做了精简和代码风...
摘要:第二种情况是箭头函数的如果指向普通函数它的继承于该普通函数。箭头函数的指向全局,使用会报未声明的错误。 showImg(https://segmentfault.com/img/remote/1460000018610072?w=600&h=400); 箭头函数是ES6的API,相信很多人都知道,因为其语法上相对于普通函数更简洁,深受大家的喜爱。就是这种我们日常开发中一直在使用的API...
摘要:箭头函数没有自己的值,箭头函数中所使用的来自于函数作用域链。使用箭头函数打印对于内层函数,它本身并没有值,其使用的来自作用域链,来自更高层函数的作用域。 《JavaScript 深入浅出》系列: JavaScript 深入浅出第 1 课:箭头函数中的 this 究竟是什么鬼? JavaScript 深入浅出第 2 课:函数是一等公民是什么意思呢? 普通函数与箭头函数 普通函数指的是...
阅读 3312·2021-11-22 14:44
阅读 2521·2019-08-30 14:10
阅读 2562·2019-08-30 13:12
阅读 1187·2019-08-29 18:36
阅读 1323·2019-08-29 16:16
阅读 3293·2019-08-26 10:33
阅读 1738·2019-08-23 18:16
阅读 359·2019-08-23 18:12