jQuery通用遍历方法each的实现

each介绍

jQuery 的 each 方法,做为一个通用遍历方法,可用于遍历对象和数组。git

语法为:github

jQuery.each(object, [callback])

回调函数拥有两个参数:第一个为对象的成员或数组的索引,第二个为对应变量或内容。数组

// 遍历数组 $.each( [0,1,2], function(i, n){ console.log( "Item #" + i + ": " + n ); }); // Item #0: 0 // Item #1: 1 // Item #2: 2
// 遍历对象 $.each({ name: "John", lang: "JS" }, function(i, n) { console.log("Name: " + i + ", Value: " + n); }); // Name: name, Value: John // Name: lang, Value: JS

退出循环

尽管 ES5 提供了 forEach 方法,可是 forEach 没有办法停止或者跳出 forEach 循环,除了抛出一个异常。可是对于 jQuery 的 each 函数,若是须要退出 each 循环可以使回调函数返回 false,其它返回值将被忽略。app

$.each( [0, 1, 2, 3, 4, 5], function(i, n){ if (i > 2) return false; console.log( "Item #" + i + ": " + n ); }); // Item #0: 0 // Item #1: 1 // Item #2: 2

初版

那么咱们该怎么实现这样一个 each 方法呢?函数

首先,咱们确定要根据参数的类型进行判断,若是是数组,就调用 for 循环,若是是对象,就使用 for in 循环,有一个例外是类数组对象,对于类数组对象,咱们依然可使用 for 循环。性能

更多关于类数组对象的知识,咱们能够查看《JavaScript专题之类数组对象与arguments》this

那么又该如何判断类数组对象和数组呢?实际上,咱们在《JavaScript专题之类型判断(下)》就讲过jQuery 数组和类数组对象判断函数 isArrayLike 的实现。spa

因此,咱们能够轻松写出初版:prototype

// 初版 function each(obj, callback) { var length, i = 0; if ( isArrayLike(obj) ) { length = obj.length; for ( ; i < length; i++ ) { callback(i, obj[i]) } } else { for ( i in obj ) { callback(i, obj[i]) } } return obj; }

停止循环

如今已经能够遍历对象和数组了,可是依然有一个效果没有实现,就是停止循环,按照 jQuery each 的实现,当回调函数返回 false 的时候,咱们就停止循环。这个实现起来也很简单:对象

咱们只用把:

callback(i, obj[i])

替换成:

if (callback(i, obj[i]) === false) { break; }

轻松实现停止循环的功能。

this

咱们在实际的开发中,咱们有时会在 callback 函数中用到 this,先举个不怎么恰当的例子:

// 咱们给每一个人添加一个 age 属性,age 的值为 18 + index var person = [ {name: 'kevin'}, {name: 'daisy'} ] $.each(person, function(index, item){ this.age = 18 + index; }) console.log(person)

这个时候,咱们就但愿 this 能指向当前遍历的元素,而后给每一个元素添加 age 属性。

指定 this,咱们可使用 call 或者 apply,其实也很简单:

咱们把:

if (callback(i, obj[i]) === false) { break; }

替换成:

if (callback.call(obj[i], i, obj[i]) === false) { break; }

关于 this,咱们再举个经常使用的例子:

$.each($("p"), function(){ $(this).hover(function(){ ... }); })

虽然咱们常常会这样写:

$("p").each(function(){ $(this).hover(function(){ ... }); })

可是由于 $("p").each() 方法是定义在 jQuery 函数的 prototype 对象上面的,而 $.each()方法是定义 jQuery 函数上面的,调用的时候不从复杂的 jQuery 对象上调用,速度快得多。因此咱们推荐使用第一种写法。

回到第一种写法上,就是由于将 this 指向了当前 DOM 元素,咱们才能使用 $(this)将当前 DOM 元素包装成 jQuery 对象,优雅的使用 hover 方法。

因此最终的 each 源码为:

function each(obj, callback) { var length, i = 0; if (isArrayLike(obj)) { length = obj.length; for (; i < length; i++) { if (callback.call(obj[i], i, obj[i]) === false) { break; } } } else { for (i in obj) { if (callback.call(obj[i], i, obj[i]) === false) { break; } } } return obj; }

性能比较

咱们在性能上比较下 for 循环和 each 函数:

var arr = Array.from({length: 1000000}, (v, i) => i); console.time('for') var i = 0; for (; i < arr.length; i++) { i += arr[i]; } console.timeEnd('for') console.time('each') var j = 0; $.each(arr, function(index, item){ j += item; }) console.timeEnd('each')

这里显示一次运算的结果:

性能比较

从上图能够看出,for 循环的性能是明显好于 each 函数的,each 函数本质上也是用的 for 循环,究竟是慢在了哪里呢?

咱们再看一个例子:

function each(obj, callback) { var i = 0; var length = obj.length for (; i < length; i++) { value = callback(i, obj[i]); } } function eachWithCall(obj, callback) { var i = 0; var length = obj.length for (; i < length; i++) { value = callback.call(obj[i], i, obj[i]); } } var arr = Array.from({length: 1000000}, (v, i) => i); console.time('each') var i = 0; each(arr, function(index, item){ i += item; }) console.timeEnd('each') console.time('eachWithCall') var j = 0; eachWithCall(arr, function(index, item){ j += item; }) console.timeEnd('eachWithCall')

这里显示一次运算的结果:

性能比较

each 函数和 eachWithCall 函数惟一的区别就是 eachWithCall 调用了 call,从结果咱们能够推测出,call 会致使性能损失,但也正是 call 的存在,咱们才能将 this 指向循环中当前的元素

相关文章
相关标签/搜索