概述

很早就想研究underscore源码了,虽然underscore.js这个库有些过时了,但是我还是想学习一下库的架构,函数式编程以及常用方法的编写这些方面的内容,又恰好没什么其它要研究的了,所以就了结研究underscore源码这一心愿吧。

underscore.js源码研究(1)

underscore.js源码研究(2)

underscore.js源码研究(3)

underscore.js源码研究(4)

underscore.js源码研究(5)

underscore.js源码研究(6)

underscore.js源码研究(7)

underscore.js源码研究(8)

参考资料:underscore.js官方注释undersercore 源码分析undersercore 源码分析 segmentfault

链式调用

对于一个对象的方法,我们可以在方法里面return this来使它支持链式调用。如果不修改原方法的源码,我们可以通过下面的函数使它支持链式调用。

//其中obj是一个对象,functions是对象的方法名数组
function chained(obj, functions) {
//由于functions是一个数组,所以可以使用foreach
functions.forEach((funcName) => {
const func = obj[funcName];
//修改原方法
obj[funcName] = function() {
func.apply(this, arguments);
return this;
}
})
}

示例如下:

let speaker = {
haha() {
console.log('haha');
},
yaya() {
console.log('yaya');
},
lala() {
console.log('lala');
}
} chained(speaker, ['haha', 'lala', 'yaya']);
speaker.haha().yaya().haha().lala().yaya();

输出如下:

haha
yaya
haha
lala
yaya

underscore.js里面的链式调用

underscore.js里面的链式调用与上面的稍微有些不同。

它的机制是这样的:

  1. 把underscore.js的方法全部挂载到_下面。
  2. 利用_.function方法遍历所有挂载在_下面的方法,然后挂载到_.prototype下面,这样生成的underscore对象就可以直接调用这些方法了。
  3. 在把方法挂载到_.prototype下面的时候,会利用类似上面的函数对方法进行改写(增加return this)。
  4. 在改写的时候建立一个标识_chain,来标识这个方法能不能链式调用。

下面来具体看看是怎么运作的:

首先,利用下面的函数,可以得到一个数组,这个数组里面全是挂载到_下面的方法名字

_function = _.methods = function(obj) {
var names = [];
for( var key in obj) {
if(_.isFunction(obj[key])) names.push(key);
}
return names.sort();
};

然后我们对每一个方法名字,把它挂载到_.prototype下面

_.mixin = function(obj) {
_.each(_.functions(obj), function(name) {
var func = _[name] = obj[name];
_.prototype[name] = function() {
var args = [this._wrapped];
push.apply(args, arguments);
return chainResult(this, func.apply(_, args));
};
});
};
_.mixin(_);

从上面可以看到,在挂载的时候返回一个chainResult方法处理过的东西。而args就等于原对象+参数1+参数2+...组成的数组,所以func.apply(_, args)就是_[func](原对象,参数1,参数2,...)处理后的结果

再来看看chainResult是怎么样的:

var chainResult = function(instance, obj){
return instance._chain ? _(obj).chain() : obj;
};

它判断如果原对象能够链式调用,那么处理后的结果obj也能够链式调用。怎么让结果也能链式调用呢?答案是使用_.chain方法:

_chain = function(obj) {
var instance = _(obj);
instance._chain = true;
return instance;
}

这个方法和我们最开始的chained方法差不多,但是它会把原对象转化为underscore对象,并且这个对象的_chain属性为真,即它能够被链式调用。

所以如果我们要在underscore.js中实现链式调用的话,直接用chain方法即可,实例如下:

_([1,2]).push(3).push(5) //输出4,并不是我们想要的
_([1,2]).chain().push(3).push(5).value() //输出[1, 2, 3, 5],正是我们想要的

可以看到,我们上面用到了value()函数,它能够中断链式调用,并不返回指针而是返回值,代码如下:

_.prototype.value = function() {
return this._wrapped;
}

等等,_wrapped又是什么?它是生成underscore对象前的原对象,看下面的代码:

var _ = function(obj) {
if(obj instanceof _) return obj;
if(!(this instanceof _)) return new _(obj);
this._wrapped = obj;
}

也就是说,在使用_生成underscore对象的时候,把原对象储存在_wrapped属性里面了,所以_wrapped就是原对象。

underscore.js源码研究(8)的更多相关文章

  1. underscore.js源码研究(7)

    概述 很早就想研究underscore源码了,虽然underscore.js这个库有些过时了,但是我还是想学习一下库的架构,函数式编程以及常用方法的编写这些方面的内容,又恰好没什么其它要研究的了,所以 ...

  2. underscore.js源码研究(6)

    概述 很早就想研究underscore源码了,虽然underscore.js这个库有些过时了,但是我还是想学习一下库的架构,函数式编程以及常用方法的编写这些方面的内容,又恰好没什么其它要研究的了,所以 ...

  3. underscore.js源码研究(5)

    概述 很早就想研究underscore源码了,虽然underscore.js这个库有些过时了,但是我还是想学习一下库的架构,函数式编程以及常用方法的编写这些方面的内容,又恰好没什么其它要研究的了,所以 ...

  4. underscore.js源码研究(4)

    概述 很早就想研究underscore源码了,虽然underscore.js这个库有些过时了,但是我还是想学习一下库的架构,函数式编程以及常用方法的编写这些方面的内容,又恰好没什么其它要研究的了,所以 ...

  5. underscore.js源码研究(3)

    概述 很早就想研究underscore源码了,虽然underscore.js这个库有些过时了,但是我还是想学习一下库的架构,函数式编程以及常用方法的编写这些方面的内容,又恰好没什么其它要研究的了,所以 ...

  6. underscore.js源码研究(2)

    概述 很早就想研究underscore源码了,虽然underscore.js这个库有些过时了,但是我还是想学习一下库的架构,函数式编程以及常用方法的编写这些方面的内容,又恰好没什么其它要研究的了,所以 ...

  7. underscore.js源码研究(1)

    概述 很早就想研究underscore源码了,虽然underscore.js这个库有些过时了,但是我还是想学习一下库的架构,函数式编程以及常用方法的编写这些方面的内容,又恰好没什么其它要研究的了,所以 ...

  8. underscore.js源码解析(五)—— 完结篇

    最近公司各种上线,所以回家略感疲惫就懒得写了,这次我准备把剩下的所有方法全部分析完,可能篇幅过长...那么废话不多说让我们进入正题. 没看过前几篇的可以猛戳这里: underscore.js源码解析( ...

  9. underscore.js 源码

    underscore.js 源码 underscore]JavaScript 中如何判断两个元素是否 "相同" Why underscore 最近开始看 underscore.js ...

随机推荐

  1. Django框架之models和不依赖Qquery的ajax请求

    一.models表字段 1)class表字段的创建 AutoField(Field) - int自增列,必须填入参数 primary_key=True BigAutoField(AutoField) ...

  2. MongDB 配置成本地服务

    一.配置mangodb. 首先现在下来直接安装就好了, 可视化工具可以后面再安装. 在自己安装的目录下面(这个随意做好能够放一块)建立一个data文件, 再在下面建立一个db文件 在cmd中进入到Mo ...

  3. rails 新建user的phonenumber字段

    1.新建字段 //rails g migration add_字段名_to_表名 字段名:字段类型 rails g migration add_title_to_contents title:stri ...

  4. 2019.01.04 洛谷P4719 【模板】动态dp(链分治+ddp)

    传送门 ddpddpddp模板题. 题意简述:给你一棵树,支持修改一个点,维护整棵树的最大带权独立集. 思路: 我们考虑如果没有修改怎么做. 貌似就是一个sbsbsb树形dpdpdp,fi,0f_{i ...

  5. C# AOP框架入门(转)

    出处:https://www.cnblogs.com/isaboy/p/Csharp_AOP_Log.html AOP面向切面编程(Aspect Oriented Programming),是通过预编 ...

  6. TCP/IP协议(6):传输层之UDP

    一. UDP用户数据报协议,它是一个无连接的,面向数据报的协议,它不提供可靠性但传输速度比TCP要快. UDP数据报中的“UDP长度”为两个字节,所以我们要发送的UDP数据最多支持65507大约68K ...

  7. 动态样式CSS

    <link>标签可以把外部css样式引入HTML页面 <style>元素用于指定嵌入的样式 通过修改link的href属性,改变引入的css样式 function loadSt ...

  8. 阿里云oss怎么上传文件夹

    最近公司在做工程项目,实现文件夹云存储上传 网上找了很久,发现很多项目都存在一些问题,但还是让我找到了一个成熟的项目. 工程: 对项目的文件夹云存储上传功能做出分析,找出文件夹上传的原理,对文件夹的云 ...

  9. CentOS 利用Yum安装mysql后无法启动(MySQL Daemon failed to start.)

    安装mysql-server [root@iZwz9cl4i8oy1reej7o8pmZ ~]# yum install -y mysql-server 进入/etc执行mysql_install_d ...

  10. 返回结点值为e的二叉树指针

    题意为,如果二叉树某结点的值为e(假定是整型二叉树),返回这个结点的指针.初看这道题,联想到二叉树可以很简单的遍历,直接返回这个指针不就行了吗?如下图所示,假如要返回值为3的结点指针: 设计好了一个函 ...