随着ES6标准的普及,JavaScript已经拥有许多新的语法糖,这让我们编写可读的,高质量的代码变得更加方便,但即使这样仍然会遇到一些潜在的"陷阱"。
箭头函数提供了更简洁和更短的语法,其中一个可用功能是您可以将函数编写为具有隐式返回值的lambda表达式。
例如:
const numbers = [1, 2, 3, 4];
numbers.map(function(n) {
return n * n;
});
可以使用箭头函数简化
const numbers = [1, 2, 3, 4];
numbers.map(n => n * n);
但如果我们希望映射到对象,可能并不会像我们期望的那样。
const numbers = [1, 2, 3, 4];
numbers.map(n => { value: n });
上面的结果并非映射到对象,而是映射到undefined。花括号被解释为箭头函数的块范围,值语句实际上最终成为标签,上面的代码将被Javascript解释器翻译成:
const numbers = [1, 2, 3, 4];
numbers.map(function(n) {
value:
n
return;
});
解决方法非常微妙,我们只需要将对象包装在括号中,将其转换为表达式而不是块语句,如下所示:
const numbers = [1, 2, 3, 4];
numbers.map(n => ({ value: n }));
箭头函数没有自己的this绑定,这意味着它们的this值将this与封闭的词法范围的值相同。尽管语法可以说是“更光滑”,但箭头函数功能并不能适用于所有的情况,比如你可能会遇到:
let calculator = {
value: 0,
add: (values) => {
this.value = values.reduce((a, v) => a + v, this.value);
},
};
calculator.add([1, 2, 3]);
console.log(calculator.value); // 0
我们期望这里的this绑定的是计算器对象,但它实际上会导致this未定义或指向全局对象。而常规函数是有一个this绑定,当在一个对象上调用时,它将指向该对象,因此在添加对象方法时仍然应该使用常规函数。
let calculator = {
value: 0,
add(values) {
this.value = values.reduce((a, v) => a + v, this.value);
},
};
calculator.add([10, 10]);
console.log(calculator.value); // 20
由于箭头函数没有this绑定,因此使用Function.prototype.call,Function.prototype.bind和Function.prototype.apply也不能给它绑定上下文对象。
比如下面的代码我们尝试使用Function.prototype.call为箭头函数绑定计算器对象,但实际上绑定的是全局对象。
const adder = {
add: (values) => {
this.value = values.reduce((a, v) => a + v, this.value);
},
};
let calculator = {
value: 0
};
adder.add.call(calculator, [1, 2, 3]);
console.log(calculator.value); // 0
虽然自动插入分号(ASI)不是一个新功能,但它是JavaScript中最奇怪的功能之一,因此值得一提。
请看下面的代码:
return
{
value: 42
}
有人可能认为它将返回对象,但它实际上将返回undefined,因为分号插入发生,使其成为一个空的return语句,后跟一个block语句和一个label语句。
最终的代码会被解释成:
return;
{
value: 42
};
所以即使不得已使用分号,也不要在括号、模板字符串和普通字符串前使用换行。
ES6中的集合不允许重复的元素,当元素是基本数据类型时,会比较值是否相等;当元素是引用数据类型时,会比较引用对象是否是同一个。
例如:
let set = new Set();
set.add([1, 2, 3]);
set.add([1, 2, 3]);
console.log(set.size); // 2
集合的最终长度是2,由于两次添加的数组不是同一个。
let set = new Set();
set.add([1, 2, 3].join(','));
set.add([1, 2, 3].join(','));
console.log(set.size); // 1
最终会得到大小为1的集合,因为字符串是不可变。如果您发现自己需要存储一组可以序列化的对象,则可以将其用作解决方法。
在JavaScript中,常规函数被提升到词法范围的顶部,这意味着下面的示例将按照人们的预期运行:
let segment = new Segment();
function Segment() {
this.x = 0;
this.y = 0;
}
但是对于类来说情况并非如此,实际上没有提升类,下面的代码会导致异常Uncaught ReferenceError: Segment is not defined
let segment = new Segment();
class Segment {
constructor() {
this.x = 0;
this.y = 0;
}
}
请看下方的代码:
(function(){
try {
return true;
} finally {
return false;
}
})() // false
我们可能认为第一个return语句使函数实际返回并弹出调用堆栈,但Finally关键字例外,因为finally语句总是运行所以结果是finally块中的return语句返回。
本文首发于公众号「前端新视界」,分享前端资讯、技术干货、编码人生,欢迎关注。
原文:https://mp.weixin.qq.com/s/mhAxXCZ4lMgD5APRN7i_qQ
30-seconds-code这个项目是一个非常优秀的JavaScript项目,这里总结了大量的使用ES6语法实现的代码块,项目的设计目标就是更简洁,更高效,更快速的实现基础代码模块。
不同版本的Node.js对Babel有不同的支持力度。为了让Node.js支持所需的ES6语法,可以加入Babel的支持。
我们通常会有意无意的把JavaScript和其他编程语言区分开,有一个重要因素是……由于它的特性本身(太灵活了吧),它似乎不仅仅是一种语言,而更像是一帮老司机在矫情造作之下乱伦出来的生态系统。
JavaScript是一种轻量性脚本语言 ,其语句以;结束,语句块用{...},js应许末尾不加,浏览器Js引擎会自动在每个语句的结尾补上,js功能主要是:动态修改html页面内容,包括创建、删除html页面元素、修改html页面元素的内容
css的顶层样式表由两种规则组成的规则列表构成,一种称为at—rule规则,也就是at规则,另一种是qualified rule,也就是普通规则。今天就学习一下at规则
export用于定义要输出的变量(let、var、const、function、class),定义的变量与值是动态绑定关系。匿名定义本质上是采用 default 为名称,与上面2个的区别是在加载时可以不用写大括号还能自定义名称。
箭头函数;扩展运算符 ...的一个通用的用法就是把对象展开;变量声明es6中不建议继续使用var来声明变量,推荐使用let和const声明,以此避免var声明存在的弊端
.htaccess文件是运行Apache Web Server的Web服务器的配置文件,对配置和重定向Apache Web Server文件系统很有用。在这里,我将讨论.htaccess文件RewriteRule语法规则。
AST 是抽象语法树的缩写词,表示编程语言的语句和表达式中生成的 token。有了 AST,解释器或编译器就可以生成机器码或者对一条指令求值。
你希望如果 response 或者 response.settings 或者 response.settings.n 不存在(值为 null 或者 undefined)时,result 保底值为 100。但是上面代码在 n 为 0 的时候,也会让 result 变成 100,你实际上希望此时 result 为 0。
内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!