写在前面:本<JQuery源码解析>系列是基于一些前辈们的文章进行进一步的分析、细化、修改而写出来的,在这边感谢那些慷慨提供科普文档的技术大拿们。

要查阅JQ的源文件请下载开发版的JQ.js文档,下载地址:http://jquery.com/download/ 注意选择其中的development版本进行下载,如下图所示

开发版本的JQ.js属于非压缩的源文件,方便我们阅读和分析其代码。 下载完用Dreamweaver或其它代码编辑器打开查阅即可。我们今后分析的代码也是基于1.11.0版本的JQ源代码。

第一眼看JQ的源代码或许会感到混乱和没头绪,特别是会卡在36行的代码那里迟迟找不到“function( window, noGlobal ) {”的后半段终点:

// Pass this if window is not defined yet
}(typeof window !== "undefined" ? window : this, function( window, noGlobal ) { /* 很容易看卡在这里,不知道function( window, noGlobal )到哪里才结束 */

不过现在可以直接告诉你,function( window, noGlobal )可谓是贯穿后面代码的全部部分,属于JQ总体架构的一部分。JQ的主体代码如下:

(function( global, factory ) {

    if ( typeof module === "object" && typeof module.exports === "object" ) {
// For CommonJS and CommonJS-like environments where a proper window is present,
// execute the factory and get jQuery
// For environments that do not inherently posses a window with a document
// (such as Node.js), expose a jQuery-making factory as module.exports
// This accentuates the need for the creation of a real window
// e.g. var jQuery = require("jquery")(window);
// See ticket #14549 for more info
module.exports = global.document ?
factory( global, true ) :
function( w ) {
if ( !w.document ) {
throw new Error( "jQuery requires a window with a document" );
}
return factory( w );
};
} else {
factory( global );
} // Pass this if window is not defined yet
}(typeof window !== "undefined" ? window : this, function( window, noGlobal ) { /* 刚刚说的容易看卡住的地方 */     //说白了这里就是写各种JQ功能函数的地方,大概有一万多行 }));

我们把上面这个主架构函数再简化成这样:

(function( global, factory ) {
.....
}(参数1,参数2)); //参数1就是那个“typeof window !== "undefined" ? window : this”,参数2就是那个“function( window, noGlobal ){..这里有一万多行哦..}”

这种函数形式我们称之为匿名函数,也就是没有函数名的函数。介绍匿名函数之前我们先来看看函数的定义方式有哪些:

第一种,也是常规的一种:

function abc(x){
alert (2 * x);
}

第二种:这种方法使用了Function构造函数,把参数列表和函数体都作为字符串,很不方便,不建议使用:

var abc = new Function('x', 'alert (2 * x);');

第三种:等号右侧先定义好函数,再赋给等号左侧的变量。

var abc = function(x) { alert( 2* x ) ; }

这里你需要知道的一点是,等号右边的函数,就是匿名函数了。我们可以用一个变量指向它,并称此变量为该函数的“函数字面量”(注意不是函数名字,匿名函数的不会偶函数名的),像var aa=function(x) { alert( 2* x ) ; } ,则“aa”就是该函数的函数字面量。我们可以通过 aa(实参) 来执行该函数。

匿名函数还有另一种定义形式:

(function(x){
alert(2* x);
})(3); /* 也可以写做
(function(x){
alert(2* x);
}(3));
*/

这种形式的函数没有函数名,却有调用参数的专属括号(像上面的是“(3)”,表示把3作为参数来强制调用赋值给x,最终alert出6)。

需要知道的是,像

(function(参数){...;}(调用参数));    //“参数”和“调用参数”都是可有可无、数量可变的

这种形式的匿名函数,都是会立即执行的,无须写什么调用函数的代码就会自动执行。你大可以写个页面文件,加上

<script language="javascript">
(function(x){
alert( 1+x );
})(2);
</script>

然后运行该页面,立刻会弹出“3”的窗口。

了解了匿名函数,那么我们回看JQ源码:

可见JQ源码就是一个匿名函数

 (function( global, factory ) {

     ... //这里是为了兼容nodejs等一些其它的js框架;

 }(a,b))

其中形参global的实参a是一个三目运算符   typeof window !== "undefined" ? window : this    用于判断当前执行环境是否支持window类型,是的话返回window,否则返回this

形参factory的实参b则是一个函数,里面包含了一万多行的JQ功能函数  function( window, noGlobal ) { ......  }

既然这个外部匿名函数的参数的值我们都清楚了,那么来看下这个匿名函数又是啥作用的?(光看JQ自带的英文注释我们可以大致知道它是为了兼容node.js、sea-JS等符合CommonJS规范或类CommonJS规范的js框架)

首先我们看这行判断语句:      if ( typeof module === "object" && typeof module.exports === "object" )

玩过node.js的朋友自然会知道module.export和export是node.js中用来创建模块的方法,那么就好理解了,若此条件成立,则要执行下面语句来兼容node.js(说白了就是利用形参factory做中间人,来把JQ的各个功能模块用node.js创建模块的方法创建起来)

 {
module.exports = global.document ? //三目运算符,先判断当前环境是否支持window.document属性
//(注意我们上面提到过形参global的实参是window) factory( global, true ) : //支持的话就好办啦,只要咱用常规的浏览器一般都是支持的,那就直接module.exports = factory( global, true ), //把JQ后面那一万多行的功能函数扩展到node.js里面。(注意我们上面提到过形参factory的实参是实现JQ各种功能的一个外部函数) function( w ) { //如果当前环境不支持window.document属性,那就写个函数扔个Error说这环境不适用JQ,但依旧返回JQ的功能函数(但大部分估计是不能用的了)
if ( !w.document ) { throw new Error( "jQuery requires a window with a document" ); } return factory( w ); };
}

嗯,这样就兼容了node.js咯,那么如果咱没有用node.js这种CommonJS规范的框架,也就是说条件if ( typeof module === "object" && typeof module.exports === "object" )不成立。那就直接执行后面else里的部分:

factory( global );

也就是直接引入JQ那一万多行的功能函数即可。

话说也写了不少字,第一部分先到这里吧 :)

JQuery源码解析(一)的更多相关文章

  1. jQuery 源码解析二:jQuery.fn.extend=jQuery.extend 方法探究

    终于动笔开始 jQuery 源码解析第二篇,写文章还真是有难度,要把自已懂的表述清楚,要让别人听懂真的不是一见易事. 在 jQuery 源码解析一:jQuery 类库整体架构设计解析 一文,大致描述了 ...

  2. jquery源码解析:代码结构分析

    本系列是针对jquery2.0.3版本进行的讲解.此版本不支持IE8及以下版本. (function(){ (21, 94)     定义了一些变量和函数,   jQuery = function() ...

  3. jquery 源码解析

    静态与实力方法共享设计 遍历方法 $(".a").each() //作为实例方法存在 $.each() //作为静态方法存在 Jquery源码 jQuery.prototype = ...

  4. jQuery源码解析资源便签

    最近开始解读jQuery源码,下面的链接都是搜过来的,当然妙味课堂 有相关的一系列视频,长达100多期,就像一只蜗牛慢慢爬, 至少品读三个框架,以后可以打打怪,自己造造轮子. 完全理解jQuery源代 ...

  5. 三.jQuery源码解析之jQuery的框架图

    这张图片是对jQuery源码截图,一点一点拼出来的. 现在根据这张图片来对jQuery框架做一些说明. 一.16~9404行可以发现,最外层是一个自调用函数.当jQuery初始化时,这个自调用函数包含 ...

  6. jquery源码解析:addClass,toggleClass,hasClass详解

    这一课,我们将继续讲解jQuery对元素属性操作的方法. 首先,我们先看一下这几个方法是如何使用的: $("#div1").addClass("box1 box2&quo ...

  7. jquery源码解析:jQuery数据缓存机制详解2

    上一课主要讲了jQuery中的缓存机制Data构造方法的源码解析,这一课主要讲jQuery是如何利用Data对象实现有关缓存机制的静态方法和实例方法的.我们接下来,来看这几个静态方法和实例方法的源码解 ...

  8. jquery源码解析:jQuery数据缓存机制详解1

    jQuery中有三种添加数据的方法,$().attr(),$().prop(),$().data().但是前面两种是用来在元素上添加属性值的,只适合少量的数据,比如:title,class,name等 ...

  9. jquery源码解析:jQuery工具方法when详解

    我们先来看when方法是如何使用的: var cb = $.when();   //when方法也是返回一个延迟对象,源码是return deferred.promise();返回的延迟对象不能修改状 ...

随机推荐

  1. Linux下高cpu占有率的调试方案

    1.用top命令查看哪个进程占用CPU高 gateway网关进程14094占用CPU高达891%,这个数值是进程内各个线程占用CPU的累加值.   2.用top -H -p pid命令查看进程内各个线 ...

  2. C# 京东模拟登录小结

    最近有需要模拟京东登录,在解决过程中遇到了一些问题,因此这里记录下来避免以后遇到同样的问题. 首先第一步需要做的是找到登录请求网址和关于请求所需的一些信息.这里可以用抓取工具或者直接用firebug或 ...

  3. DEV设计之自动流水号,DEV专家解答,自己折腾了半天也没有搞定,怪英文不好

    () 老外专家给了回答,结果没有全到懂,又折腾了20分钟朋友提示才搞定 获取一个自动增加1的流水号值, 第一个参数是本事的数据库连接对象,第2个参数是也这个值为唯一标识返回来一个增量的值,第三个好像没 ...

  4. caffe net 可视化工具

    http://ethereon.github.io/netscope/#/editor 将对应的网络输入到里面,然后按shift+enter即可查看对应的网络结构

  5. 188. Best Time to Buy and Sell Stock IV leetcode解题笔记

    Say you have an array for which the ith element is the price of a given stock on day i. Design an al ...

  6. Test注解的两个属性(转)

    Test注解的两个属性:expected和timeout Junit的Test注解支持两个可选的参数expected和timeout.expected声明一个测试方法必须抛出一个异常.如果不抛出异常或 ...

  7. 深入理解php底层:php生命周期 [转]

    1.PHP的运行模式: PHP两种运行模式是WEB模式.CLI模式.无论哪种模式,PHP工作原理都是一样的,作为一种SAPI运行. 1.当我们在终端敲入php这个命令的时候,它使用的是CLI. 它就像 ...

  8. Excel导入-----导出(包含所选和全部)操作

    在做系统的时候,很多时候信息量太大,这时候就需要进行Excel表格信息的导入和导出,今天就来给大家说一下我使用Excel表格信息导入和导出的心得. 1:首先需要在前端显示界面View视图中添加导入Ex ...

  9. 高尔夫管理系统SSH

    登录-----------http://localhost:8080/GOLF/Denglu 一.Action 1.处理今日消费数据逻辑的 package com.chinasofti.golf.ac ...

  10. PWM波控制舵机总结

    文章转自:http://www.geek-workshop.com/thread-70-1-1.html 一.关于舵机: 舵机(英文叫Servo):它由直流电机.减速齿轮组.传感器和控制电路组成的一套 ...