JavaScript小技巧

1、Every 函数 和 some 函数的巧妙使用

every函数和some函数并非是所有开发人员都熟悉的函数。但是,它们在某些情况下非常有用。让我们从every函数开始。如果你想知道数组的所有元素是否都通过特定测试,则可以使用此函数。本质上,这是在遍历数组的每个元素并检查它们是否全部正确。

这听起来可能有点抽象,所以让我们看下面的示例。它并不像听起来那样复杂。

const random_numbers = [ 13, 2, 37, 18, 5 ]
const more_random_numbers = [ 0, -1, 30, 22 ]
const isPositive = (number) => {return number > 0}
random_numbers.every(isPositive); // returns true
more_random_numbers.every(isPositive); // returns false

每个函数都返回一个布尔值。如果数组中的所有元素均通过测试,则将返回true。如果数组中的元素之一未通过测试,则将返回false。

如果你愿意,也可以使用匿名函数作为测试器函数:

random_numbers.every((number) => {return number > 0})

some功能几乎与every功能完全相同。只有一个主要区别:some函数测试数组中至少一个元素是否通过实现的测试。

如果我们看一下前面的示例,并使用some函数而不是every函数,则两个数组都将返回true,因为两个数组都包含一个正数。

const random_numbers = [ 13, 2, 37, 18, 5 ]
const more_random_numbers = [ 0, -1, 30, 22 ]
const isPositive = (number) => {return number > 0}
random_numbers.some(isPositive); // returns true
more_random_numbers.some(isPositive); // returns true

2、有条件地设置一个变量

有条件地设置变量既容易,又可以使代码看起来更优雅。应用此技巧时无需编写if语句-就我个人而言,这是我最喜欢的JavaScript技巧之一。

那么如何有条件地设置变量?

const timezone = user.preferred_timezone || 'America/New_York'

在上面的示例中,我们检查用户是否具有首选时区。如果用户具有首选时区,我们将使用该时区。如果用户没有首选时区,我们将使用默认时区,即“ America / New_York”。

与使用if语句相比,此代码看起来干净得多。

let timezone = 'America/New_York'
if (user.preferred_timezone) { timezone = user.preferred_timezone}

这样,看起来是不是更加整洁?

3、在数组中强制转换值

有时你想将所有值都转换为数组。例如,当你使用三元等于运算符检查数组中是否存在某个数字时,就是其中一种情况。

我最近遇到了一个多重选择问题。select选项的HTML值是字符串而不是整数,这很有意义。所选值的数组如下所示:​​​​​​​

let selected_values = [ '1', '5', '8' ]

问题是我正在检查所选值的数组中是否存在某个整数。没有成功。我用了一个使用三元等于运算符的相交函数。既然'5'!== 5,我必须找到一个解决方案。

我认为,最漂亮的解决方案是将数组中的所有值都转换为整数。尝试执行此操作时,我偶然发现了一个痛苦而又简单却优雅的解决方案。

selected_values = selected_values.map(Number) // [ 1, 5, 8 ]

除了将所有值都强制转换为整数,还可以通过简单地更改map函数的参数将数组中的所有值强制转换为布尔值。​​​​​​​

selected_values = selected_values.map(Boolean)

4、对象解构

一旦了解了对象解构的知识,你可能每天都会使用它。

但是,到底什么是解构?

解构是一种JavaScript表达式,可让你从数组,对象,映射和集合中提取数据到它们自己的变量中。它允许你一次从多个对象中提取对象的属性或从数组中提取项目。

让我们看下面的示例,其中有一个用户对象。如果要将用户名存储在变量中,则必须在新行中将其分配给变量。而且,如果要将性别也存储在变量中,则必须再次执行相同操作。​​​​​​​

const user = {name: 'Frank',age: 23,gender: 'M',member: false}
const name = user.nameconst gender = user.gender

通过解构,你可以使用以下语法直接获取对象属性的变量:​​​​​​​

const { name, age, gender, member } = user;
console.log(name) // Frank
console.log(age) // 23
console.log(gender) // M
console.log(member) // false

5、更好的调试使用性能

开发人员要做的一件重要的事就是调试。但是,调试不只是使用console.log在控制台中打印一堆日志消息。

你是否知道控制台对象是分析代码段性能的好方法?但是,大多数开发人员仅了解调试其代码的标准console.log方法。

控制台对象具有更多有用的功能。它具有time and timeEnd函数,可以帮助你分析性能。它真的很简单。

在要测试的代码前面,调用console.time函数。此函数有一个参数,该参数采用描述你要分析的内容的字符串。在要测试的代码末尾,调用console.timeEnd函数。你为该函数提供与第一个参数相同的字符串。然后,你将在控制台中查看运行代码所花费的时间。​​​​​​​

console.time('loop') 
for (let i = 0; i < 10000; i++) { // Do stuff here } 
console.timeEnd('loop')

6、过滤唯一值

Set对象类型是在ES6中引入的,配合展开操作...一起,我们可以使用它来创建一个新数组,该数组只有唯一的值。

const array = [1, 1, 2, 3, 5, 5, 1]
const uniqueArray = [...new Set(array)];
console.log(uniqueArray); // Result: [1, 2, 3, 5]

在ES6之前,隔离惟一值将涉及比这多得多的代码。

此技巧适用于包含基本类型的数组:undefined,null,boolean,string和number。(如果你有一个包含对象,函数或其他数组的数组,你需要一个不同的方法!)

7、 与或运算

三元运算符是编写简单(有时不那么简单)条件语句的快速方法,如下所示:

x > 100 ? 'Above 100' : 'Below 100';
x > 100 ? (x > 200 ? 'Above 200' : 'Between 100-200') : 'Below 100';

但有时使用三元运算符处理也会很复杂。相反,我们可以使用'与'&&和'或'|| 逻辑运算符以更简洁的方式书写表达式。这通常被称为“短路”或“短路运算”。

它是怎么工作的

假设我们只想返回两个或多个选项中的一个。

使用&&将返回第一个条件为假的值。如果每个操作数的计算值都为true,则返回最后一个计算过的表达式。​​​​​​​

let one = 1, two = 2, three = 3;
console.log(one && two && three); // Result: 3
console.log(0 && null); // Result: 0

使用||将返回第一个条件为真的值。如果每个操作数的计算结果都为false,则返回最后一个计算过的表达式。​​​​​​​

let one = 1, two = 2, three = 3;
console.log(one || two || three); // Result: 1
console.log(0 || null); // Result: null

例一

假设我们想返回一个变量的长度,但是我们不知道变量的类型。

我们可以使用if/else语句来检查foo是可接受的类型,但是这可能会变得非常冗长。或运行可以帮助我们简化操作:​​​​​​​

return (foo || []).length

如果变量foo是true,它将被返回。否则,将返回空数组的长度:0。

例二

你是否遇到过访问嵌套对象属性的问题?你可能不知道对象或其中一个子属性是否存在,这可能会导致令人沮丧的错误。

假设我们想在this.state中访问一个名为data的属性,但是在我们的程序成功返回一个获取请求之前,data 是未定义的。

根据我们使用它的位置,调用this.state.data可能会阻止我们的应用程序运行。为了解决这个问题,我们可以将其做进一步的判断:​​​​​​​

if (this.state.data) {
   return this.state.data;
} else { 
   return 'Fetching Data';
}

但这似乎很重复。'或' 运算符提供了更简洁的解决方案:​​​​​​​

return (this.state.data || 'Fetching Data');

8、转换为布尔值

除了常规的布尔值true和false之外,JavaScript还将所有其他值视为 ‘truthy’ 或‘falsy’。

除非另有定义,否则 JavaScript 中的所有值都是'truthy',除了 0,“”,null,undefined,NaN,当然还有false,这些都是'falsy'

我们可以通过使用负算运算符轻松地在true和false之间切换。它也会将类型转换为“boolean”。

const isTrue = !0;
const isFalse = !1;
const alsoFalse = !!0;
console.log(isTrue); // Result: true
console.log(typeof true); // Result: "boolean"

9、 转换为字符串

要快速地将数字转换为字符串,我们可以使用连接运算符+后跟一组空引号""。​​​​​​​

const val = 1 + "";
console.log(val); // Result: "1"
console.log(typeof val); // Result: "string"

10、转换为数字

使用加法运算符+可以快速实现相反的效果。​​​​​​​

let int = "15";
int = +int;
console.log(int); // Result: 15
console.log(typeof int); Result: "number"

这也可以用于将布尔值转换为数字,如下所示​​​​​​​

console.log(+true); // Return: 1 
console.log(+false); // Return: 0

在某些上下文中,+将被解释为连接操作符,而不是加法操作符。当这种情况发生时(你希望返回一个整数,而不是浮点数),您可以使用两个波浪号:~~。

连续使用两个波浪有效地否定了操作,因为— ( — n — 1) — 1 = n + 1 — 1 = n。换句话说,~—16 等于15。​​​​​​​

const int = ~~"15"
console.log(int); // Result: 15
console.log(typeof int); Result: "number"

虽然我想不出很多用例,但是按位NOT运算符也可以用在布尔值上:~true = -2和~false = -1。

11、性能更好的运算

从ES7开始,可以使用指数运算符**作为幂的简写,这比编写Math.pow(2, 3) 更快。这是很简单的东西,但它之所以出现在列表中,是因为没有多少教程更新过这个操作符。​​​​​​​

console.log(2 ** 3); // Result: 8

这不应该与通常用于表示指数的^符号相混淆,但在JavaScript中它是按位异或运算符。

在ES7之前,只有以2为基数的幂才存在简写,使用按位左移操作符<<​​​​​​​

Math.pow(2, n);
2 << (n - 1);
2**n;

例如,2 << 3 = 16等于2 ** 4 = 16。

12、快速浮点数转整数

如果希望将浮点数转换为整数,可以使用Math.floor()、Math.ceil()或Math.round()。但是还有一种更快的方法可以使用|(位或运算符)将浮点数截断为整数。​​​​​​​

console.log(23.9 | 0); // Result: 23
console.log(-23.9 | 0); // Result: -23

|的行为取决于处理的是正数还是负数,所以最好只在确定的情况下使用这个快捷方式。

如果n为正,则n | 0有效地向下舍入。如果n为负数,则有效地向上舍入。更准确地说,此操作将删除小数点后面的任何内容,将浮点数截断为整数。

你可以使用~~来获得相同的舍入效果,如上所述,实际上任何位操作符都会强制浮点数为整数。这些特殊操作之所以有效,是因为一旦强制为整数,值就保持不变。

删除最后一个数字

按位或运算符还可以用于从整数的末尾删除任意数量的数字。这意味着我们不需要使用这样的代码来在类型之间进行转换。​​​​​​​

let str = "1553";
Number(str.substring(0, str.length - 1));

相反,按位或运算符可以这样写:​​​​​​​

console.log(1553 / 10 | 0) // Result: 155
console.log(1553 / 100 | 0) // Result: 15
console.log(1553 / 1000 | 0) // Result: 1

13、类中的自动绑定

我们可以在类方法中使用ES6箭头表示法,并且通过这样做可以隐含绑定。这通常会在我们的类构造函数中保存几行代码,我们可以愉快地告别重复的表达式,例如this.myMethod = this.myMethod.bind(this)​​​​​​​

import React, { Component } from React;
export default class App extends Compononent { 
    constructor(props) {     
        super(props);
        this.state = {}; 
    }
    myMethod = () => { 
        // This method is bound implicitly! 
    }
    render() { 
        return ( <> <div> {this.myMethod()} </div> </> ) 
    }
};

14、数组截断

如果要从数组的末尾删除值,有比使用splice()更快的方法。

例如,如果你知道原始数组的大小,您可以重新定义它的length属性,就像这样​​​​​​

let array = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9];
array.length = 4;
console.log(array); // Result: [0, 1, 2, 3]

这是一个特别简洁的解决方案。但是,我发现slice()方法的运行时更快。如果速度是你的主要目标,考虑使用:​​​​​​

let array = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9];
array = array.slice(0, 4);
console.log(array); // Result: [0, 1, 2, 3]

15、 获取数组中的最后一项

数组方法slice()可以接受负整数,如果提供它,它将接受数组末尾的值,而不是数组开头的值。​​​​​​​

let array = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9];
console.log(array.slice(-1)); // Result: [9]
console.log(array.slice(-2)); // Result: [8, 9]
console.log(array.slice(-3)); // Result: [7, 8, 9]

16、格式化JSON代码

最后,你之前可能已经使用过JSON.stringify,但是您是否意识到它还可以帮助你缩进JSON?

stringify()方法有两个可选参数:一个replacer函数,可用于过滤显示的JSON和一个空格值。​​​​​​​

console.log(JSON.stringify({ alpha: 'A', beta: 'B' }, null, '\t'));
// Result:
// '{
// "alpha": A,
// "beta": B
// }'

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值