虽然最近工作中没有怎么用 zepto ,但是据说 zepto 的源码比较简单,而且网上的资料也比较多,所以我就挑了 zepto 下手,希望能为以后阅读其他框架的源码打下基础吧。

源码版本

本文阅读的源码为zepto1.2.0

阅读zepto之前需要了解 javascript 原型链和闭包的知识,推荐阅读王福朋的这篇文章:深入理解 Javascript 原型和闭包,写得很详细,也非常易于阅读。

源码结构整体结构

varZepto=(function(){...})()window.Zepto=Zeptowindow.$===undefined&&(window.$=Zepto)

如果在编辑器中将 zepto 的源码折叠起来,看到的就跟上面的代码一样。

zepto 的核心是一个闭包,加载完毕后立即执行。然后暴露给全局变量zepto,如果$没有定义,也将$赋值为zepto

核心结构

在这部分中,我们先不关注 zepto 的具体实现,只看核心的结构,因此我将zepto中的逻辑先移除,得出如下的核心结构:

varzepto={},$functionZ(doms){varlen=doms.lengthfor(vari=0;i<len;i++){this[i]=doms[i]}this.length=doms.length}zepto.Z=function(doms){returnnewZ(doms)}zepto.init=function(doms){vardoms=['domObj1','domObj2','domObj3']returnzepto.Z(doms)}$=function(){returnzepto.init()}$.fn={constructor:zepto.Z,method:function(){returnthis}}zepto.Z.prototype=Z.prototype=$.fnreturn$

在源码中,可以看出,$其实是一个函数,同时在$身上又挂了很多属性和方法(这里体现在$.fn身上,其他的会在后续的文章中谈到)。

我们在使用 zepto 的时候,会用$去获取dom,并且在这些dom对象身上都有 zepto 定义的各种各样的操作方法。

从上面的伪代码中,可以看到,$其实调用了zepto.init()方法,在init方法中,会获取到dom元素集合,然后将集合交由zepto.Z()方法处理,而zepto.Z方法返回的是函数Z的一个实例。

函数Z会将doms展开,变成实例的属性,key为对应domObj的索引, 并且设置实例的length属性。

zepto.Z.prototype = Z.prototype = $.fn

读到这里,你可能会有点疑惑,$最终返回的是Z函数的实例,但是Z函数明明没有dom的操作方法啊,这些操作方法都定义在$.fn身上,为什么$可以调用这些方法呢?

其实关键在于这句代码Z.prototype = $.fn,这句代码将Zprototype指向$.fn,这样,Z的实例就继承了$.fn的方法。

既然这样就已经让Z的实例继承了$.fn的方法,那zepto.Z.prototype = $.fn又是为什么呢?

如果我们再看源码,会发现有这样的一个方法:

zepto.isZ=function(object){returnobjectinstanceofzepto.Z}

这个方法是用来判读一个对象是否为 zepto 对象,这是通过判断这个对象是否为zepto.Z的实例来完成的,因此需要将zepto.ZZprototype指向同一个对象。isZ方法会在init中用到,后面也会介绍。