Javascript Tricks

在这篇文章中将整理一些平时用到的js小技巧。这些小技巧可能在你的实际工作中或许能帮助你解决一些问题。

使用!!操作符转换布尔值

有时候我们需要对一个变量检查其是否存在或者检查值是否有一个有效值,如果存在就返回true值。为了做这样的验证,我们可以使用!!操作符来实现是非常的方便与简单。对于变量可以使用!!variable做检测,只要变量的值为:0null" "undefined或者NaN都将返回的是false,反之返回的是true。比如下面的示例:

1
2
3
4
5
6
7
8
9
10
11
function Account(cash) {
this.cash = cash;
this.hasMoney = !!cash;
}
var account = new Account(100.50);
console.log(account.cash); // 100.50
console.log(account.hasMoney); // true

var emptyAccount = new Account(0);
console.log(emptyAccount.cash); // 0
console.log(emptyAccount.hasMoney); // false

在这个示例中,只要account.cash的值大于0,那么account.hasMoney返回的值就是true

使用+将字符串转换成数字

这个技巧非常有用,其非常简单,可以交字符串数据转换成数字,不过其只适合用于字符串数据,否则将返回NaN,比如下面的示例:

1
2
3
4
5
function toNumber(strNumber) {
return +strNumber;
}
console.log(toNumber("1234")); // 1234
console.log(toNumber("ACB")); // NaN

这个也适用于Date,在本例中,它将返回的是时间戳数字:

1
console.log(+new Date()) // 1466586198892

并条件符

如果你有一段这样的代码:

1
2
3
if (conected) {
login();
}

你也可以将变量简写,并且使用&&和函数连接在一起,比如上面的示例,可以简写成这样:

1
conected && login();

如果一些属性或函数存在于一个对象中,你也可以这样做检测,如下面的代码所示:

1
user && user.login();

使用||运算符

在ES6中有默认参数这一特性。为了在老版本的浏览器中模拟这一特性,可以使用||操作符,并且将将默认值当做第二个参数传入。如果第一个参数返回的值为false,那么第二个值将会认为是一个默认值。如下面这个示例:

1
2
3
4
5
6
7
8
9
10
11
function User(name, age) {
this.name = name || "Oliver Queen";
this.age = age || 27;
}
var user1 = new User();
console.log(user1.name); // Oliver Queen
console.log(user1.age); // 27

var user2 = new User("Barry Allen", 25);
console.log(user2.name); // Barry Allen
console.log(user2.age); // 25

在循环中缓存array.length

这个技巧很简单,这个在处理一个很大的数组循环时,对性能影响将是非常大的。基本上,大家都会写一个这样的同步迭代的数组:

1
2
3
for(var i = 0; i < array.length; i++) {
console.log(array[i]);
}

如果是一个小型数组,这样做很好,如果你要处理的是一个大的数组,这段代码在每次迭代都将会重新计算数组的大小,这将会导致一些延误。为了避免这种现象出现,可以将array.length做一个缓存:

1
2
3
4
var len = array.length;
for(var i = 0; i < len; i++) {
console.log(array[i]);
}

或者,这么写:

1
2
3
for(var i = 0, len = array.length; i < len; i++) {
console.log(array[i]);
}

检测对象中属性

当你需要检测一些属性是否存在,避免运行未定义的函数或属性时,这个小技巧就显得很有用。如果你打算定些一些跨兼容的浏览器代码,你也可能会用到这个小技巧。例如,你想使用document.querySelector()来选择一个id,并且让它能兼容IE6浏览器,但是在IE6浏览器中这个函数是不存在的,那么使用这个操作符来检测这个函数是否存在就显得非常的有用,如下面的示例:

1
2
3
4
5
6
7
8
9
10
if ('querySelector' in document) {
document.querySelector("#id");
} else {
document.getElementById("id");
}

var oEvent = {
start : "ontouchstart" in document ? 'touchstart' : 'mousedown',
end : "ontouchend" in document ? 'touchend' : 'mouseup'
};

在这个示例中,如果document不存在querySelector函数,那么就会调用docuemnt.getElementById("id")

如果documentontouchstart属性(或函数),使用touchstart,否则使用mousedown

获取数组中最后一个元素

Array.prototype.slice(begin, end)用来获取beginend之间的数组元素。如果你不设置end参数,将会将数组的默认长度值当作end值。但有些同学可能不知道这个函数还可以接受负值作为参数。如果你设置一个负值作为begin的值,那么你可以获取数组的最后一个元素。如:

1
2
3
4
var array = [1,2,3,4,5,6];
console.log(array.slice(-1)); // [6]
console.log(array.slice(-2)); // [5,6]
console.log(array.slice(-3)); // [4,5,6]

复制数组

复制数组有三种方法:

  1. 通过Array.prototype.slice()方法,该方法因为不对原数组进行操作,可以通过传入参数0或者不传的方式复制;
  2. 通过Array.prototype.concat()方法,该方法通过不传参来实现数组的合并以实现复制;
  3. 最笨的方法,循环遍历老数组,再一个个push到新数组中;
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//方法1:
var arr = [3,2,1],
newArr;
newArr = arr.slice();

//方法2:
var arr = [3,2,1],
newArr;
newArr = arr.concat();

//方法3:
var arr = [3,2,1];
newArr = [];
for(var i = 0, len = arr.length; i < len; i++) {
newArr.push(arr[i])
}

测试是否是新数组可以使用arr.sort()后,检测新数组是否也被排序。

数组截断

这个小技巧主要用来锁定数组的大小,如果用于删除数组中的一些元素来说,是非常有用的。
例如,你的数组有10个元素,但你只想只要前五个元素,那么你可以通过array.length = 5来截断数组。如下面这个示例:

1
2
3
4
5
var array = [1,2,3,4,5,6];
console.log(array.length); // 6
array.length = 3;
console.log(array.length); // 3
console.log(array); // [1,2,3]

替换所有

String.replace()函数允许你使用字符串或正则表达式来替换字符串,本身这个函数只替换第一次出现的字符串,不过你可以使用正则表达多中的/g来模拟replaceAll()函数功能:

1
2
3
var string = "john john";
console.log(string.replace(/hn/, "ana")); // "joana john"
console.log(string.replace(/hn/g, "ana")); // "joana joana"

合并数组

如果你要合并两个数组,一般情况之下你都会使用concat()函数:

1
2
3
var array1 = [1,2,3];
var array2 = [4,5,6];
console.log(array1.concat(array2)); // [1,2,3,4,5,6];

NodeListarguments转换成数组

如果你运行document.querySelectorAll('p')函数时,它可能返回DOM元素的数组,也就是NodeList对象。

但这个对象不具有数组的函数功能,比如sort()reduce()map()filter()等。

为了让这些原生的数组函数功能也能用于其上面,需要将节点列表转换成数组。可以使用[].slice.call(elements)来实现:

1
2
3
var elements = document.querySelectorAll("p"); // NodeList
var arrayElements = [].slice.call(elements); // Now the NodeList is an array
var arrayElements = Array.from(elements); // This is another way of converting NodeList to Array

另外一个更有说服力的例子就是函数的arguments

1
2
3
4
5
6
7
8
//检测arguments是否为数组
function testargs(){
var arr=[1,2,3];
console.log(typeof arguments.slice);
console.log(typeof arr.slice);
console.log(arguments.constructor == Array.constructor, arr.constructor)
}
testargs();

输出:

1
2
3
undefined
function
false Array() { [native code] }

如何转换arguments

1
2
3
4
5
function arg2arr(){
var arr = Array.prototype.slice.call(arguments); // or [].slice.call(arguments, 0)
console.log(arr);
}
arg2arr(1,2,3); //[1, 2, 3]

数组元素的洗牌

对于数组元素的洗牌,不需要使用任何外部的库,比如Lodash,只要这样做:

1
2
3
4
5
6
var list = [1,2,3];
var result = list.sort(function() {
return Math.random() - 0.5
})

console.log(result); // [2,1,3]

转自:http://www.w3cplus.com/javascript/12-extremely-useful-hacks-for-javascript.html

Powered by Hexo and Hexo-theme-hiker

Copyright © 2013 - 2019 FE blog All Rights Reserved.

访客数 : | 访问量 :