Zepto源码分析之一(代码结构及初始化)
关于读源码,读jQuery自然是不错,但太过于庞大不易解读,对于小白,最好从Zepto,Lodash这样的小库入手。
这里使用的是zepto1.1.6版本为例。
自执行函数
在阅读之前,先弄清楚闭包和自执行函数
两种方式: (function() {})() 和 (function() {}())
(function() {
console.log('这里直接执行')
})()
(function () {
console.log('这里直接执行')
}())
自执行函数的优势在于,避免了全局变量的污染,即在自执行函数体内声明的变量,外部是无法访问到的。
(function () {
let val = '123'
})() console.log(val) // val is not defined
如需要获取变量val的值,只能在函数体内部返回,然后将函数赋值给一个全局变量
let temp= (function () {
let val = '123';
return val; // 必须return, 否则获取不到值
})() window.temp = temp; console.log(temp) //
而关于闭包,简单理解就是在自执行函数内部声明变量或方法,在自执行函数内就形成了一个闭合的上下文环境,外部是无法直接访问的。
源码结构
;let Zepto = (function () {
let $, zepto = {}; return $; // 最终返回值
})() window.Zepto = Zepto;
window.$ === undefined && (window.$ = Zepto);
首先声明一个字支持函数并赋给变量Zepto, 再将变量Zepto赋给全局变量window.Zepto
如果$未被声明,则也将Zepto赋给window.$全局变量
此时,外部访问Zepto或者$都可以返回自执行函数内部的$(即 return $)
再看$到底是什么
$返回一个函数,传入的参数是selector选择器(即tagName,id,或className),context暂为空。
;let Zepto = (function () {
let $, zepto = {};
$ = function(selector, context) {
console.log('获取节点');
return zepto.init(selector, context);
};
return $;
})(); window.Zepto = Zepto;
window.$ === undefined && (window.$ = Zepto);
此时,如果访问节点, 会输出'获取节点'
<div>
<span>测试</span>
<span>测试</span>
<span>测试</span>
</div>
let span = $('span');
console.log(span)
这里报错是因为init方法还未定义;
继续创建 init方法
init函数是绑定在zepto对象上的方法(之前已经声明一个zepto空对象)
zepto.init = function(selector, context) {
let dom;
// 处理dom的代码
return zepto.Z(dom, selector);
};
init处理dom部分,分为几种情况
1 . 当不传参数 $(), 直接返回zepto.Z()不传参
if (!selector) return zepto.Z()
2 . 当传入字符串参数,又分为几种
else if (typeof selector == 'string') {
selector = selector.trim() if (selector[0] == '<' && fragmentRE.test(selector))
dom = zepto.fragment(selector, RegExp.$1, context), selector = null else if (context !== undefined) return $(context).find(selector) else dom = zepto.qsa(document, selector) //这里的qsa其实就是document.querySelectorAll()
}
(1) $(' span ')
首先将清除字符串前后空格 selector.trim()
(2) $('<div></div>')
如果第一个字符是<,并且正则fragmentRE成立,
则使用fragment方法处理dom节点
if (selector[0] == '<' && fragmentRE.test(selector))
dom = zepto.fragment(selector, RegExp.$1, context), selector = null
(3) $('span', 'div')
如果传入第二个参数,即之前的context不等于undefined,则相当于执行$('div').find('span')
else if (context !== undefined) return $(context).find(selector)
若div中存在span则返回span,否则返回空数组。
(4)如以上三种不满足则执行最后一步,zepto.qsa()
3 . 如果传入函数,用document调用一个zepto对象,然后$.ready()方法, ready方法和上面的find方法一样在后面创建。
else if (isFunction(selector)) return $(document).ready(selector);
4 . 若已经是一个zepto对象,则直接返回该对象,不必再调用$(), 即 $($('span')) === $('span')
else if (zepto.isZ(selector)) return selector
zepto.isZ = function(object) {
return object instanceof zepto.Z
}
5 . 若是引用类型
else {
// 如参数是数组,则调用compact方法, 会将null或undefined剔除
if (isArray(selector)) dom = compact(selector)
// 如是DOM节点,则数组化
else if (isObject(selector))
dom = [selector], selector = null
// 以下和上面字符串参数的重复,何意未知。
else if (fragmentRE.test(selector))
dom = zepto.fragment(selector.trim(), RegExp.$1, context), selector = null
//
//
else if (context !== undefined) return $(context).find(selector)
//
else dom = zepto.qsa(document, selector)
}
init是一个初始化方法,调用init时去执行zepto.Z()函数
zepto.Z = function(dom, selector) {
dom = dom || []
dom.__proto__ = $.fn;
dom.selector = selector || '';
console.log(dom)
console.log(selector)
return dom;
};
关于Z()
1 . 传入dom,如果未传则返回[]空数组
如果不传DOM节点 $(), 则返回一个空值。
2 . 如果有dom $('span'),则
3 . 同时在该数组的原型对象上创建$.fn对象
在未定义$.fn之前,dom.__proto__原型对象上的方法是Array构造函数的内置方法
而这里使用了dom.__proto__ = $.fn 重新定义了原型对象上的方法
$.fn = {
get: function () {
console.log('自定义get方法')
},
on: function () {
console.log('自定义on方法')
}
}
可以看出,通过原型对象上创建新方法后,原来的内置方法都没有了。
如果只是想追加方法,则应该在原型对象上添加属性。
dom.__proto__.get = function() {}
dom.__proto__.on = function() {}
4 . 最后返回该DOM节点对象。此时$('span')对象就可以调用方法get和on了。
Zepto源码分析之一(代码结构及初始化)的更多相关文章
- Vue.js 源码分析(一) 代码结构
关于Vue vue是一个兴起的前端js库,是一个精简的MVVM.MVVM模式是由经典的软件架构MVC衍生来的,当View(视图层)变化时,会自动更新到ViewModel(视图模型),反之亦然,View ...
- jQuery 源码分析(一) 代码结构
jQuery是一个Javascript库,它支持链式操作方式,即对发生在同一个JQuery对象上的一组动作,可以直接接连写无需要重复获取对象.这一特点使得JQuery的代码无比优雅,而且有强大的选择器 ...
- Zepto源码分析(一)核心代码分析
本文只分析核心的部分代码,并且在这部分代码有删减,但是不影响代码的正常运行. 目录 * 用闭包封装Zepto * 开始处理细节 * 正式处理数据(获取选择器选择的DOM) * 正式处理数据(添加DOM ...
- 一个普通的 Zepto 源码分析(二) - ajax 模块
一个普通的 Zepto 源码分析(二) - ajax 模块 普通的路人,普通地瞧.分析时使用的是目前最新 1.2.0 版本. Zepto 可以由许多模块组成,默认包含的模块有 zepto 核心模块,以 ...
- Zepto源码分析(二)奇淫技巧总结
Zepto源码分析(一)核心代码分析 Zepto源码分析(二)奇淫技巧总结 目录 * 前言 * 短路操作符 * 参数重载(参数个数重载) * 参数重载(参数类型重载) * CSS操作 * 获取属性值的 ...
- MySQL源码分析以及目录结构 2
原文地址:MySQL源码分析以及目录结构作者:jacky民工 主要模块及数据流经过多年的发展,mysql的主要模块已经稳定,基本不会有大的修改.本文将对MySQL的整体架构及重要目录进行讲述. 源码结 ...
- MySQL源码分析以及目录结构
原文地址:MySQL源码分析以及目录结构作者:jacky民工 主要模块及数据流经过多年的发展,mysql的主要模块已经稳定,基本不会有大的修改.本文将对MySQL的整体架构及重要目录进行讲述. 源码结 ...
- 一个普通的 Zepto 源码分析(三) - event 模块
一个普通的 Zepto 源码分析(三) - event 模块 普通的路人,普通地瞧.分析时使用的是目前最新 1.2.0 版本. Zepto 可以由许多模块组成,默认包含的模块有 zepto 核心模块, ...
- 一个普通的 Zepto 源码分析(一) - ie 与 form 模块
一个普通的 Zepto 源码分析(一) - ie 与 form 模块 普通的路人,普通地瞧.分析时使用的是目前最新 1.2.0 版本. Zepto 可以由许多模块组成,默认包含的模块有 zepto 核 ...
- ffplay源码分析2-数据结构
ffplay是FFmpeg工程自带的简单播放器,使用FFmpeg提供的解码器和SDL库进行视频播放.本文基于FFmpeg工程4.1版本进行分析,其中ffplay源码清单如下: https://gith ...
随机推荐
- python输出
学习之前普及一个知识,在python2.X的版本中是不可以输入中文的,如果一定要输入中文就必须要在整段代码的最上面第一行备注一下:# _*_ coding:utf-8 _*_ print函数,这是一个 ...
- UOJ#172. 【WC2016】论战捆竹竿 字符串 KMP 动态规划 单调队列 背包
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ172.html 题解 首先,这个问题显然是个背包问题. 然后,可以证明:一个字符串的 border 长度可 ...
- Git基本操作指令
Git是世界上目前最先进的分布式版本控制系统. 工作原理图: Workspace工作区,Index暂存区,Repository本地仓库区,Remote远程仓库. SVN与Git的最主要的区别? SVN ...
- Ubuntu中eclipse端口被占
我的eclipse有时候会闪退,然后再次打开运行HTML程序的时候会提示端口被占,即使之前改过端口也会提示,然后我就在网上搜索解决办法,目前亲测有效的是 输入以下命令: lsof -i:8888 // ...
- 一道dfs和dp结合的好题 --- Longest Run on a SnowboardUVA-10285
题目链接: https://vjudge.net/problem/19213/origin 大致题意: 一个滑雪者想知道自己在固定高度的山坡中最多能滑的距离是多少. 思路: 首先想到的就是dfs,但是 ...
- 数字图像特征提取之HOG特征
1.灰度化:(以便可以使用sobel等算子计算梯度)2.gamma校正: (降低光照影响)3.求每个像素的梯度和方向: (利用任意一种梯度算子,例如:sobel,laplacian等,对该patch进 ...
- office图标(word,powerpoint,excel)异常(变成白板)问题修复
都是wps搞的鬼. 如果先装了wps后卸载wps再装office就可能出现这个问题. 终于解决了!!先装个wps 2016,进入设置→高级→兼职性关联.这时你会发现ppt文件图标不是白板了.重启打开w ...
- Python开发实战PDF
Python开发实战(高清版)PDF 百度网盘 链接:https://pan.baidu.com/s/1iP9VmwuzDMfdZTfpupR3CA 提取码:a523 复制这段内容后打开百度网盘手机A ...
- Urozero Autumn 2016. BAPC 2016
A. Airport Logistics 根据光路最快原理以及斯涅尔定律,可以得到从定点$P$进入某条直线的最佳入射角. 求出每个端点到每条线段的最佳点,建图求最短路即可. 时间复杂度$O(n^2\l ...
- TypeError: 'module' object is not callable
pkuseg.py 内容如下: import pkusegseg = pkuseg.pkuseg()text = seg.cut('我爱北京天安门')print(text) 原因是py文件名于包名一样 ...