underscore.js源码解析(三)
最近工作比较忙,做不到每周两篇了,周末赶着写吧,上篇我针对一些方法进行了分析,今天继续。
没看过前两篇的可以猛戳这里:
underscore.js源码GitHub地址: https://github.com/jashkenas/underscore/blob/master/underscore.js本文解析的underscore.js版本是1.8.3
_.map/_.collect
_.map = _.collect = function(obj, iteratee, context) {
iteratee = cb(iteratee, context);
//判断是否是数组,是的话就将里面的属性取出
var keys = !isArrayLike(obj) && _.keys(obj),
length = (keys || obj).length,
results = Array(length);
for (var index = 0; index < length; index++) {
var currentKey = keys ? keys[index] : index;
results[index] = iteratee(obj[currentKey], currentKey, obj);
}
return results;
};
map和each的区别就是map是将最后的结果以数组的形式返回
createReduce
var createReduce = function(dir) {
var reducer = function(obj, iteratee, memo, initial) {
//判断是否是数组
var keys = !isArrayLike(obj) && _.keys(obj),
length = (keys || obj).length,
//判断迭代方向,dir为1是从左向右迭代,dir为-1是从右向左迭代(从_.reduce和_.reduceRight里就可以清晰的看出来)
index = dir > 0 ? 0 : length - 1;
//判断是否存在初始值
if (!initial) {
//如果没有初始值,则将第一个设置为初始值,前面判断了是否是数组,是数组返回数组否则返回对象
memo = obj[keys ? keys[index] : index];
//根据方向将初始位置赋给index,为了下边遍历使用
index += dir;
}
//进行遍历
for (; index >= 0 && index < length; index += dir) {
var currentKey = keys ? keys[index] : index;
//进行迭代运算
memo = iteratee(memo, obj[currentKey], currentKey, obj);
}
return memo;
};
return function(obj, iteratee, memo, context) {
//通过参数数量判断是否有初始值
var initial = arguments.length >= 3;
//调用reducer()
return reducer(obj, optimizeCb(iteratee, context, 4), memo, initial);
};
};
看了上面的注释你可以清晰的发现createReduce的结构已经很清晰了,内部调用reducer函数,先判断有没有初始值,如果没有则根据迭代方向给初始值赋值,然后进行循环的迭代。
createPredicateIndexFinder
var createPredicateIndexFinder = function(dir) {
return function(array, predicate, context) {
//迭代函数
predicate = cb(predicate, context);
var length = getLength(array);
//判断遍历方向,dir为1是从左向右,dir为-1是从右向左
var index = dir > 0 ? 0 : length - 1;
for (; index >= 0 && index < length; index += dir) {
//遍历返回符合条件的是第几个
if (predicate(array[index], index, array)) return index;
}
//否则返回-1
return -1;
};
};
_.findIndex/_.findLastIndex
_.findIndex = createPredicateIndexFinder(1);
_.findLastIndex = createPredicateIndexFinder(-1);
调用上面的createPredicateIndexFinder内部函数,两者只是遍历的方向不同,最终返回相应的索引
_.findKey
_.findKey = function(obj, predicate, context) {
//迭代函数
predicate = cb(predicate, context);
var keys = _.keys(obj), key;
for (var i = 0, length = keys.length; i < length; i++) {
//获取对象属性
key = keys[i];
if (predicate(obj[key], key, obj)) return key;
}
};
返回满足条件的属性
_.find / _.detect
_.find = _.detect = function(obj, predicate, context) {
var key;
if (isArrayLike(obj)) {
//如果是数组通过findIndex获取满足条件的索引
key = _.findIndex(obj, predicate, context);
} else {
//如果是对象则通过findKey获取满足条件的属性
key = _.findKey(obj, predicate, context);
}
//根据上面取到的索引或属性,返回相应的值
if (key !== void 0 && key !== -1) return obj[key];
};
_.filter/_.select
_.filter = _.select = function(obj, predicate, context) {
var results = [];
predicate = cb(predicate, context);
//遍历数据,将符合条件的存入results数组当中,并返回
_.each(obj, function(value, index, list) {
if (predicate(value, index, list)) results.push(value);
});
return results;
};
_.negate
_.negate = function(predicate) {
return function() {
return !predicate.apply(this, arguments);
};
};
_.negate就是一个取反的函数
_.reject
_.reject = function(obj, predicate, context) {
return _.filter(obj, _.negate(cb(predicate)), context);
};
_.every/_.all
_.every = _.all = function(obj, predicate, context) {
predicate = cb(predicate, context);
//判断数组还是对象
var keys = !isArrayLike(obj) && _.keys(obj),
length = (keys || obj).length;
for (var index = 0; index < length; index++) {
//数组获取索引,对象获取属性
var currentKey = keys ? keys[index] : index;
//如果有不满足条件的就返回false
if (!predicate(obj[currentKey], currentKey, obj)) return false;
}
return true;
};
_.every就是判断是否所有的数据都满足条件
_.some
_.some = _.any = function(obj, predicate, context) {
predicate = cb(predicate, context);
var keys = !isArrayLike(obj) && _.keys(obj),
length = (keys || obj).length;
for (var index = 0; index < length; index++) {
var currentKey = keys ? keys[index] : index;
if (predicate(obj[currentKey], currentKey, obj)) return true;
}
return false;
};
跟every的判断结构差不多,只不过最后的判断条件不同,some是判断时候有满足条件的,有的话就返回true
_.values
_.values = function(obj) {
var keys = _.keys(obj);
var length = keys.length;
var values = Array(length);
for (var i = 0; i < length; i++) {
values[i] = obj[keys[i]];
}
return values;
};
_.values就是讲obj的所有值拷贝到数组当中
_.sortedIndex
_.sortedIndex = function(array, obj, iteratee, context) {
iteratee = cb(iteratee, context, 1);
var value = iteratee(obj);
var low = 0, high = getLength(array);
while (low < high) {
var mid = Math.floor((low + high) / 2);
if (iteratee(array[mid]) < value) low = mid + 1; else high = mid;
}
return low;
};
_.sortedIndex看过源码就可以看出是二分法查找
createIndexFinder
var createIndexFinder = function(dir, predicateFind, sortedIndex) {
return function(array, item, idx) {
var i = 0, length = getLength(array);
if (typeof idx == 'number') {
//判断方向,1是从前到后,-1则为从后到前
if (dir > 0) {
i = idx >= 0 ? idx : Math.max(idx + length, i);
} else {
length = idx >= 0 ? Math.min(idx + 1, length) : idx + length + 1;
}
//如果是排序好的就使用二分法
} else if (sortedIndex && idx && length) {
idx = sortedIndex(array, item);
//判断找出的值是否一样,是就返回这个值,否则返回-1
return array[idx] === item ? idx : -1;
}
if (item !== item) {
//对item为NaN的处理
idx = predicateFind(slice.call(array, i, length), _.isNaN);
return idx >= 0 ? idx + i : -1;
}
for (idx = dir > 0 ? i : length - 1; idx >= 0 && idx < length; idx += dir) {
//通过遍历的方法找出item对应的索引
if (array[idx] === item) return idx;
}
//找不到则返回-1
return -1;
};
};
_.indexOf/_.lastIndexOf
_.indexOf = createIndexFinder(1, _.findIndex, _.sortedIndex);
_.lastIndexOf = createIndexFinder(-1, _.findLastIndex);
_.contains/_.includes/_.include
//判断是否包含对应的值
_.contains = _.includes = _.include = function(obj, item, fromIndex, guard) {
//如果是对象,则将obj的所有值拷贝到数组当中
if (!isArrayLike(obj)) obj = _.values(obj);
if (typeof fromIndex != 'number' || guard) fromIndex = 0;
//查找是否存在这个值,如果存在,indexOf 返回相应的索引,则为true,如果不存在,indexOf 返回-1,则为false
return _.indexOf(obj, item, fromIndex) >= 0;
};
_.invoke
//对每一个元素都执行一次方法,最后把结果存入数组返回
_.invoke = restArgs(function(obj, method, args) {
var isFunc = _.isFunction(method);
return _.map(obj, function(value) {
//如果是函数则每个元素都执行一遍方法,如果不是,则返回所有的值,最后结果以数组的形式返回
var func = isFunc ? method : value[method];
return func == null ? func : func.apply(value, args);
});
});
小结
今天分析的几个内部函数理解起来有点难度,理解要多看几遍分析不同的情况,也许结合调用的例子理解起来会容易一点,分析我都写在了代码间的注释里,这样读起源码会容易一点,今天就写到这里吧,有点累了,明天还要上班,剩下的方法在之后的文章里都会分析到。
感谢大家的观看,也希望能够和大家互相交流学习,有什么分析的不对的地方欢迎大家批评指出
参考资料
https://segmentfault.com/a/1190000000531871 http://www.w3cfuns.com/house/17398/note/class/id/bb6dc3cabae6651b94f69bbd562ff370underscore.js源码解析(三)的更多相关文章
- underscore.js源码解析(五)—— 完结篇
最近公司各种上线,所以回家略感疲惫就懒得写了,这次我准备把剩下的所有方法全部分析完,可能篇幅过长...那么废话不多说让我们进入正题. 没看过前几篇的可以猛戳这里: underscore.js源码解析( ...
- underscore.js源码解析(四)
没看过前几篇的可以猛戳这里: underscore.js源码解析(一) underscore.js源码解析(二) underscore.js源码解析(三) underscore.js源码GitHub地 ...
- underscore.js源码解析(二)
前几天我对underscore.js的整体结构做了分析,今天我将针对underscore封装的方法进行具体的分析,代码的一些解释都写在了注释里,那么废话不多说进入今天的正文. 没看过上一篇的可以猛戳这 ...
- underscore.js源码解析(一)
一直想针对一个框架的源码好好的学习一下编程思想和技巧,提高一下自己的水平,但是看过一些框架的源码,都感觉看的莫名其妙,看不太懂,最后找到这个underscore.js由于这个比较简短,一千多行,而且读 ...
- underscore.js源码解析【'_'对象定义及内部函数】
(function() { // Baseline setup // -------------- // Establish the root object, `window` (`self`) in ...
- underscore.js源码解析【对象】
// Object Functions // ---------------- // Keys in IE < 9 that won't be iterated by `for key in . ...
- underscore.js源码解析【数组】
// Array Functions // --------------- // Get the first element of an array. Passing **n** will retur ...
- underscore.js源码解析【集合】
// Collection Functions // -------------------- // The cornerstone, an `each` implementation, aka `f ...
- underscore.js源码解析【函数】
// Function (ahem) Functions // ------------------ // Determines whether to execute a function as a ...
随机推荐
- Servlet基础知识总结
Servlet是JavaWeb应用开发的核心组件.Servlet运行在Servlet容器中(例如最常用的Tomcat),它可以为各种客户请求提供相应服务.Servlet可以轻松完成以下任务: 动态生成 ...
- koa2学习笔记03 - 给koa2配置session ——koa2结构分层、配置数据库、接口
前言 这一章写的很没有底气,因为我完全不懂一个正经的后台应用是怎么结构分层的, 所有只能按照我自己的理解去写,即使这样也仅仅只分离出了controller层, 至于所谓的service层,dao层,完 ...
- PHP目前比较常见的五大运行模式
做 php 开发的应该都知道 php 运行模式概念吧,本文将要和大家分享的是关于php目前比较常见的五大运行模式:包括cgi .fast-cgi.cli.isapi.apache模块的DLL ,下面作 ...
- Zeta--S3 Linux抓取一帧YUV图像后使用硬件编码器编码成H.264
#include <stdio.h> #include <stdlib.h> #include <string.h> #include <getopt.h&g ...
- 【非原创】ISBN码
#include<stdio.h>int main(void){ char a[14],mod[12]="0123456789X"; #include <stdi ...
- php安装redis拓展
1. 查看是否安装redis库 查看是否安装redis库了.可以通过下面2种方式查看. phpinfo()是否能输出redis的加载信息 在命令行执行`php -m` 输出gd 2. 安装redis库 ...
- Flex4中的拖动技术
下面列一个最简单的例子,在Flex中,拖动原来如此简单 ? 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 <?xm ...
- 20155229 2016-2017-2《Java程序设计》课程总结
20155229 2016-2017-2<Java程序设计>课程总结 每周作业链接汇总 预备作业1:对专业的期待和对师生关系的理解 预备作业2:分析自我技能延展到c语言学习状况 预备作业3 ...
- 2017-2018-1 20155315 《信息安全系统设计基础》加分作业:实现mypwd
学习pwd命令 man pwd查看 pwd命令用于显示当前工作目录,是Linux下最常用的命令之一.在不太确定当前位置时,就会使用pwd来判定当前目录在文件系统内的确切位置. 环境变量OLDPWD表示 ...
- 成都Uber优步司机奖励政策(4月11日)
滴快车单单2.5倍,注册地址:http://www.udache.com/ 如何注册Uber司机(全国版最新最详细注册流程)/月入2万/不用抢单:http://www.cnblogs.com/mfry ...