当前位置:首页 > 日记 > 正文

jQuery源码分析之init的详细介绍

jQuery源码分析之init的详细介绍

init 构造器

由于这个函数直接和 jQuery() 的参数有关,先来说下能接受什么样的参数。

源码中接受 3 个参数:

init: function (selector, context, root) { ...}
  1. jQuery() ,空参数,这个会直接返回一个空的 jQuery 对象,return this。
  2. jQuery( selector [, context ] ) ,这是一个标准且常用法,selector 表示一个 css 选择器,这个选择器通常是一个字符串,#id 或者 .class 等,context 表示选择范围,即限定作用,可为 DOM,jQuery 对象。
  3. jQuery( element|elements ) ,用于将一个 DOM 对象或 DOM 数组封装成 jQuery 对象。
  4. jQuery( jQuery object|object ) ,会把普通的对象或 jQuery 对象包装在 jQuery 对象中。
  5. jQuery( html [, ownerDocument ] ) ,这个方法用于将 html 字符串先转成 DOM 对象后在生成 jQuery 对象。
  6. jQuery( html, attributes ) ,和上一个方法一样,不过会将 attributes 中的方法和属性绑定到生成的 html DOM 中,比如 class 等。
  7. jQuery( callback ) ,此方法接受一个回掉函数,相当于 window.onload 方法,只是相对于。

介绍完入口,就开始来看源码。

init: function (selector, context, root) { var match, elem; // 处理: $(""), $(null), $(undefined), $(false) if (!selector) { return this; } // rootjQuery = jQuery( document ); root = root || rootjQuery; // 处理 HTML 字符串情况,包括 $("<div>")、$("#id")、$(".class") if (typeof selector === "string") { //此部分拆分,留在后面讲 // HANDLE: $(DOMElement) } else if (selector.nodeType) { this[0] = selector; this.length = 1; return this; // HANDLE: $(function) } else if (jQuery.isFunction(selector)) { return root.ready !== undefined ? root.ready(selector) : // Execute immediately if ready is not present selector(jQuery); } return jQuery.makeArray(selector, this);}

上面有几点需要注意,root = root || rootjQuery; ,这个参数在前面介绍用法的时候,就没有提及,这个表示 document,默认的话是 rootjQuery,而 rootjQuery = jQuery( document )

可以看出,对于处理 $(DOMElement) ,直接是把 jQuery 当作一个数组,this[0] = DOMElement 。其实,这要从 jQuery 的基本构造讲起,我们完成一个 $('div.span') 之后,然后一个 jQuery 对象(this),其中会得到一组(一个)DOM 对象,jQuery 会把这组 DOM 对象当作数组元素添加过来,并给一个 length。后面就像一些链式函数操作的时候,若只能对一个 DOM 操作,比如 width、height,就只对第一个元素操作,若可以对多个 DOM 操作,则会对所有 DOM 进行操作,比如 css()。

jQuery 大题思路如下,这是一个非常简单点实现:

jQuery.prototype = { // 简单点,假设此时 selector 用 querySelectorAll init: function(selector){ var ele = document.querySelectorAll(selector); // 把 this 当作数组,每一项都是 DOM 对象 for(var i = 0; i < ele.length; i++){ this[i] = ele[i]; } this.length = ele.length; return this; }, //css 若只有一个对象,则取其第一个 DOM 对象 //若 css 有两个参数,则对每一个 DOM 对象都设置 css css : function(attr,val){ for(var i = 0; i < this.length; i++){ if(val == undefined){ if(typeof attr === 'object'){ for(var key in attr){ this.css(key, attr[key]); } }else if(typeof attr === 'string'){ return getComputedStyle(this[i])[attr]; } }else{ this[i].style[attr] = val; } } },}

所以对于 DOMElement 的处理,直接将 DOM 赋值给数组后,return this。

jQuery.makeArray 是一个绑定 数组的函数,和上面的原理一样,后面会谈到。

在介绍下面的内容之前,先来介绍一个 jQuery 中一个识别 Html 字符串的正则表达式,

var rquickExpr = /^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]+))$/;rquickExpr.exec('<div>') //["<div>", "<div>", undefined]rquickExpr.exec('<div></div>') //["<div></div>", "<div></div>", undefined]rquickExpr.exec('#id') //["#id", undefined, "id"]rquickExpr.exec('.class') //null

上面这一系列的正则表达式 exec,只是为了说明 rquickExpr 这个正则表达式执行后的结果,首先,如果匹配到,结果数组的长度是 3,如果匹配到 <div> 这种 html,数组的第三个元素是 underfined,如果匹配到 #id,数组的第二个元素是 underfined,如果匹配不到,则为 null。

另外还有一个正则表达式:

var rsingleTag = ( /^<([a-z][^\/\0>:\x20\t\r\n\f]*)[\x20\t\r\n\f]*\/?>(?:<\/\1>|)$/i );rsingleTag.test('<div></div>') //truersingleTag.test('<div ></div>') //truersingleTag.test('<div class="cl"></div>') //falsersingleTag.test('<div></ddiv>') //false

这个正则表达式主要是对 html 的字符串进行验证,达到不出差错的效果。在这里不多介绍 exec 和正则表达式了。

下面来看下重点的处理 HTMl 字符串的情况:

if (selector[0] === "<" && selector[selector.length - 1] === ">" && selector.length >= 3) { // 这个其实是强行构造了匹配 html 的情况的数组 match = [null, selector, null];} else { match = rquickExpr.exec(selector);}// macth[1] 限定了 html,!context 对 #id 处理if (match && (match[1] || !context)) { // HANDLE: $(html) -> $(array) if (match[1]) { //排除 context 是 jQuery 对象情况 context = context instanceof jQuery ? context[0] : context; // jQuery.merge 是专门针对 jQuery 合并数组的方法 // jQuery.parseHTML 是针对 html 字符串转换成 DOM 对象 jQuery.merge(this, jQuery.parseHTML( match[1], context && context.nodeType ? context.ownerDocument || context : document, true)); // HANDLE: $(html, props) if (rsingleTag.test(match[1]) && jQuery.isPlainObject(context)) { for (match in context) { // 此时的 match 非彼时的 match if (jQuery.isFunction(this[match])) { this[match](context[match]); // ...and otherwise set as attributes } else { this.attr(match, context[match]); } } } return this; // 处理 match(1) 为 underfined 但 !context 的情况 } else { elem = document.getElementById(match[2]); if (elem) { // this[0] 返回一个标准的 jQuery 对象 this[0] = elem; this.length = 1; } return this;}// 处理一般的情况,find 实际上上 Sizzle,jQuery 已经将其包括进来,下章详细介绍// jQuery.find() 为 jQuery 的选择器,性能良好} else if (!context || context.jquery) { return (context || root).find(selector);// 处理 !context 情况} else { // 这里 constructor 其实是 指向 jQuery 的 return this.constructor(context).find(selector);}

关于 nodeType,这是 DOM 的一个属性,详情 Node.nodeType MDN。nodeType 的值一般是一个数字,比如 1 表示 DOM,3 表示文字等,也可以用这个值是否存在来判断 DOM 元素,比如 context.nodeType。

整个 init 函数等构造逻辑,非常清晰,比如 (selector, context, root) 三个参数,分别表示选择的内容,可能存在的的限制对象或 Object,而 root 则默认的 jQuery(document) 。依旧采用 jQuery 常用的方式,对每一个变量的处理都非常的谨慎。

如果仔细看上面两部分源代码,我自己也加了注释,应该可以把整个过程给弄懂。

find 函数实际上是 Sizzle,已经单独出来一个项目,被在 jQuery 中直接使用,将在下章介绍 jQuery 中的 Sizzle 选择器。通过源码,可以发现:

jQuery.find = function Sizzle(){...}jQuery.fn.find = function(selector){ ... //引用 jQuery.find jQuery.find() ...}

衍生的函数

init 函数仍然调用了不少 jQuery 或 jQuery.fn 的函数,下面来逐个分析。

jQuery.merge

这个函数通过名字,就知道它是用来干什么的,合并。

jQuery.merge = function (first, second) { var len = +second.length, j = 0, i = first.length; for (; j < len; j++) { first[i++] = second[j]; } first.length = i; return first;}

这样子就可以对类似于数组且有 length 参数的类型进行合并,我感觉主要还是为了方便对 jQuery 对象的合并,因为 jQuery 对象就是有 length 的。

jQuery.parseHTML

这个函数也非常有意思,就是将一串 HTML 字符串转成 DOM 对象。

首先函数接受三个参数,第一个参数 data 即为 html 字符串,第二个参数是 document 对象,但要考虑到浏览器的兼容性,第三个参数 keepScripts 是为了删除节点里所有的 script tags,但在 parseHTML 里面没有体现,主要还是给 buildFragment 当作参数。

总之返回的对象,是一个 DOM 数组或空数组。

jQuery.parseHTML = function (data, context, keepScripts) { if (typeof data !== "string") { return []; } // 平移参数 if (typeof context === "boolean") { keepScripts = context; context = false; } var base, parsed, scripts; if (!context) { // 下面这段话的意思就是在 context 缺失的情况下,建立一个 document 对象 if (support.createHTMLDocument) { context = document.implementation.createHTMLDocument(""); base = context.createElement("base"); base.href = document.location.href; context.head.appendChild(base); } else { context = document; } } // 用来解析 parsed,比如对 "<div></div>" 的处理结果 parsed:["<div></div>", "div"] // parsed[1] = "div" parsed = rsingleTag.exec(data); scripts = !keepScripts && []; // Single tag if (parsed) { return [context.createElement(parsed[1])]; } // 见下方解释 parsed = buildFragment([data], context, scripts); if (scripts && scripts.length) { jQuery(scripts).remove(); } return jQuery.merge([], parsed.childNodes);}

buildFragment 函数主要是用来建立一个包含子节点的 fragment 对象,用于频发操作的添加删除节点。parsed = buildFragment([data], context, scripts);建立好一个 fragment 对象,用 parsed.childNodes 来获取这些 data 对应的 HTML。

jQueyr.makeArray

jQuery 里面的函数调用,真的是一层接一层,虽然有时候光靠函数名,就能知道这函数的作用,但其中思考之逻辑还是挺参考意义的。

jQuery.makeArray = function (arr, results) { var ret = results || []; if (arr != null) { if (isArrayLike(Object(arr))) { jQuery.merge(ret, typeof arr === "string" ? [arr] : arr); } else { push.call(ret, arr); } } return ret;}

makeArray 把左边的数组或字符串并入到右边的数组或一个新数组,其中又间接的引用 jQuery.merge 函数。

接下来是着 isArrayLike 函数,可能需要考虑多方面的因素,比如兼容浏览器等,就有了下面这一长串:

function isArrayLike(obj) { // Support: real iOS 8.2 only (not reproducible in simulator) // `in` check used to prevent JIT error (gh-2145) // hasOwn isn't used here due to false negatives // regarding Nodelist length in IE var length = !!obj && "length" in obj && obj.length, type = jQuery.type(obj); if (type === "function" || jQuery.isWindow(obj)) { return false; } return type === "array" || length === 0 || typeof length === "number" && length > 0 && (length - 1) in obj;}

总结

这篇文章主要介绍了jQuery中比较重要的入口函数,之后将会继续讲解 Sizzle,jQuery 中的选择器。感兴趣的朋友们请继续关注,谢谢大家的支持。

相关文章

JS字符串false转boolean的方法 |

JS字符串false转boolean的方法 |

方法,推荐,字符串,电脑软件,JS,大家都知道在JS的世界里, 0、-0、null、""、false、undefined 或 NaN,这些都可以自动转化为布尔的 false,那么字符串的"false"是不是false呢,答案是否定的,if("false") 来判断的话,是等于true的所以今天遇到…

怎么在excel中使用矩阵函数在excel

怎么在excel中使用矩阵函数在excel

函数,方法,教程,矩阵,电脑软件,  矩阵函数:定义域和值域都属于方阵的函数称为矩阵函数,它有多种定义方法。当用方阵幂级数的和函数来定义矩阵函数时,方阵函数f(a)=,其中自变量a和函数值f(a)都是n阶方阵,Ck是常系数。下面小编教你在excel中使用…

详谈Angular路由与Nodejs路由的区

详谈Angular路由与Nodejs路由的区

路由,区别,电脑软件,Angular,Nodejs,1、觉得angualr.js的路由是针对于单页面的路由,每次路由发生变化,只是页面的状态发生变化,页面本身没有发生跳转2、express的路由是针对多页面的,也就是说他做的页面,路由的切换是伴随着页面的切换3、所以建…

wps打开表格出错怎么处理

wps打开表格出错怎么处理

处理方法,方法,表格,怎么处理,电脑软件,  我们有时在打开wps表格的时候,会弹出打开错误的提示窗口,如果出现这样的问题,那么,我们应该如何解决呢?下面就让小编告诉你解决wps表格打开出错的方法,希望小编整理的学习资料对大家有用。解决wps表格…

powerpoint预设动画在哪个栏目

powerpoint预设动画在哪个栏目

方法,设置,动画,切换动画,栏目,  在制作幻灯片的时候,要怎么去设置页面切换动画呢?其实方法很简单,但是新手不会,怎么办?有简单易懂的方法吗?下面给大家分享ppt怎么设置页面切换动画的方法啦,希望小编整理的资料对大家有帮助。ppt设置页面切…

怎么用Word2017统计字数word2017怎

怎么用Word2017统计字数word2017怎

统计,步骤,字数,电脑软件,strong,  使用Word2017自己打了一篇文档以后,自己想要查看一下自己打了多少文字,怎么快速的查看文字个数呢,使用过Word低版本的朋友有些朋友知道怎么查看字数,但是版本升级以后,查看的方法有些区别,但是还是不影响到小…

如何设置word2013自动生成目录

如何设置word2013自动生成目录

自动生成,如何设置,目录,电脑软件,  word2013中的目录用起来比word2010的更顺手,也更舒服。word2013插入目录用起来也不难,却能让整个文档阅读起来更方便,那么下面就由小编给大家分享下让word2013自动生成目录的技巧,希望能帮助到您。自动生…

JavaScript原生节点操作小结

JavaScript原生节点操作小结

节点操作,原生,电脑软件,JavaScript,前言:原生是Javascript的基础,还是需要多多重视,时间长都忘记了,现在整理一下。获取子节点children 不是标准的dom属性,但是几乎被所有浏览器支持。不包含文本节点.注意:在IE中,children包含注释节点。childNod…

excel表格间隔求和的方法excel表格

excel表格间隔求和的方法excel表格

方法,间隔,表格,电脑软件,excel,  Excel中的表格具体该如何间隔求和呢?下面是由小编分享的excel表格间隔求和的方法,以供大家阅读和学习。excel表格间隔求和的方法表格间隔求和步骤1:我们看到图中的颜色标记,我们是隔行求和,每隔一行求和一次…

jQuery动画_动力节点节点Java学院

jQuery动画_动力节点节点Java学院

节点,动画,学院,动力,电脑软件,用JavaScript实现动画,原理非常简单:我们只需要以固定的时间间隔(例如,0.1秒),每次把DOM元素的CSS样式修改一点(例如,高宽各增加10%),看起来就像动画了。但是要用JavaScript手动实现动画效果,需要编写非常复杂的代码。如…

前端框架学习总结之Angular、React

前端框架学习总结之Angular、React

学习,前端框架,详解,电脑软件,React,近几年前端的技术发展很快,细分下来,主要可以分成四个方面: 1.开发语言技术,主要是ES6&7,coffeescript,typescript等; 2.开发框架,如Angular,React,Vue.js,Angular2等; 3.开发工具的丰富和前端工…

ppt2007设置定时循环播放

ppt2007设置定时循环播放

设置,循环播放,方法,电脑软件,  平时使用课件时需要PPT以幻灯片的方式放映图片,这种放映方式在PPT上能够自由设置比图片软件好得多。那么如何ppt2007设置定时循环播放呢?接下来小编带大家了解一下吧。ppt2007设置定时循环播放的方法先打开…