(function (global, factory) {
     factory(global);
 })(this, function (window, noGlobal) {
     var rootjQuery;
     var class2type = {};
     var toString = class2type.toString;
     var arr = [];
     var indexOf = arr.indexOf;
     var myJQuery = function (selector, context) {
         return new myJQuery.fn.init(selector, context);
     };

     myJQuery.fn = myJQuery.prototype = {
         constructor: myJQuery,//为撒指定
         length: 0,
         each: function (callback) {
             return myJQuery.each(this, callback);
         },
         pushStack: function (elems) {
             var ret = myJQuery.merge(this.constructor(), elems);
             ret.prevObject = this;
             ret.context = this.context;
             return ret;
         }
     };

     myJQuery.extend = myJQuery.fn.extend = function () {
         var options, name, src, copy, target = arguments[0] || {}, length = arguments.length, i = 1;
         if (i === length) {
             target = this;
             i--;
         }
         for (; i < length; i++) {
             if ((options = arguments[i]) != null) {
                 for (name in options) {
                     src = target[name];
                     copy = options[name];
                     if (target === copy) {
                         continue;
                     }

                     if (copy !== undefined) {
                         target[name] = copy;
                     }
                 }
             }
         }
         return target;
     };

     function isArrayLike(obj) {
         var length = !!obj && "length" in obj && obj.length,
             type = myJQuery.type(obj);
         if (type === "function" || myJQuery.isWindow(obj)) {
             return false;
         }

         return type === "array" || length === 0 ||
             typeof length === "number" && length > 0 && ( length - 1 ) in obj;
     }

     myJQuery.extend({
         isWindow: function( obj ) {
             return obj != null && obj === obj.window;
         },
         //extend each,是一种静态方法 调用方式 myJQuery.each(.....)
         each: function (obj, callback) {
             var length, i = 0;
             if (isArrayLike(obj)) {
                 //如果是数组
                 length = obj.length;//设置的属性
                 for (; i < length; i++) {
                     if (callback.call(obj[i], i, obj[i]) === false) {
                         break;
                     }
                 }
             }
             else{
                 //如果是对象
                 for(i in obj){
                     if(callback.call(obj[i],i,obj[i])===false){
                         break;
                     }
                 }
             }
             return obj;
         },
         isFunction: function (obj) {
             return myJQuery.type(obj) === "function";
         },
         type: function (obj) {
             if (obj == null) {
                 return obj + "";
             }

             // Support: Android<4.0, iOS<6 (functionish RegExp)
             /*return typeof obj === "object" || typeof obj === "function" ?
              class2type[toString.call(obj)] || "object" :
              typeof obj;*/
             return typeof obj;
         },
         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;
         },
         inArray: function (elem, arr, i) {
             return arr == null ? -1 : indexOf.call(arr, elem, i);
         }
     });

     myJQuery.fn.extend({
         find: function (selector) {
             var ret = [];
             var arr = [12, 35, 68];
             ret = this.pushStack(arr);
             ret.selector = selector;
             return ret;
         }
     });

     var rnotwhite = (/\S+/g); //大s 非空白字符 小s 空白字符
     ///g 表示该表达式将用来在输入字符串中查找所有可能的匹配,返回的结果可以是多个。如果不加/g最多只会匹配一个

     function createOptions(options) {
         var object = {};
         //match 匹配成功,返回数组,否则 返回null
         myJQuery.each(options.match(rnotwhite) || [], function (_, flag) {
             object[flag] = true;
         });
         return object;
     }

     myJQuery.Callbacks = function (options) {
         options = typeof  options === "string" ? createOptions(options) : myJQuery.extend({}, options);
         var firing, memory, fired, locked, list = [], queue = [], firingIndex = -1;
         var fire = function () {
             locked = options.once;
             for (; queue.length; firingIndex = -1) {
                 memory = queue.shift();
                 while (++firingIndex < list.length) {
                     if(list[firingIndex].apply(memory[0], memory[1])===false && options.stopOnFalse===true){
                         firingIndex=list.length;
                         memory = false; //预防 参数里有 memory
                     }
                 }
             }

             //memory是一个全局变量,一直存在,而在add的时候,又有
             // if(memory && !firing){ fire(); }
             if(!options.memory){
                 memory=false;
             }

             if (locked) {
                 if (memory) {
                     list=[];
                 } else {
                     list = "";
                 }
             }
         };
         var self = {
             add: function () {
                 if (list) {
                     if (memory && !firing) {
                         firingIndex = list.length - 1;
                         queue.push(memory);
                     }

                     (function addSelf(args) {
                         myJQuery.each(args, function (_, arg) {
                             if (myJQuery.isFunction(arg)) {
                                 if (!options.unique || !self.has(arg)) {
                                     list.push(arg);
                                 }
                             } else if (arg && arg.length && myJQuery.type(arg) !== "string") {
                                 //适用于 add([fn1,fn2]);
                                 addSelf(arg);
                             }
                         });
                     })(arguments);

                     if(memory && !firing){
                         //传入 memory关键字,在add的时候,将之前所有的fn fire 一次
                         fire();
                     }
                     return this;
                 }
             },
             remove: function () {
                 myJQuery.each(arguments, function (_, arg) {
                     var index = 0;
                     while ((index = myJQuery.inArray(arg, list, index)) > -1) {
                         list.splice(index, 1);
                     }
                 });
                 return this;
             },
             has: function (fn) {
                 return fn ? myJQuery.inArray(fn, list) > -1 : list.length > 0;
             },
             empty: function () {
                 if (list) {
                     list = [];
                 }
                 return this;
             },
             disable: function () {

             },
             disabled: function () {

             },
             lock: function () {

             },
             locked: function () {

             },
             fireWith: function (context, args) {
                 //when options.once=true,the locked=true,and not fired
                 if (!locked) {
                     args = args || [];
                     queue.push([context, args]);
                     fire();
                 }
                 return this;
             },
             fire: function () {
                 self.fireWith(this, arguments);
                 return this;
             },
             fired: function () {

             }
         };
         return self;
     }
     var init = myJQuery.fn.init = function (selector, context, root) {
         if (!selector) {
             return this;
         }
         root = root || rootjQuery;
         //处理标签 div p li
         if (typeof selector === "string") {
             return ( context || root ).find(selector);
         } else if (selector.nodeType) {
             this.context = this[0] = selector;
             this.length = 1;
             return this;
         }
     };

     init.prototype = myJQuery.fn;

     rootjQuery = myJQuery(document);

     if (!noGlobal) {
         window.myJQuery = window.F$ = myJQuery;
     }

     return myJQuery;
 });

JQuery源码解析--callbacks的更多相关文章

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

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

  2. JQuery源码解析(一)

    写在前面:本<JQuery源码解析>系列是基于一些前辈们的文章进行进一步的分析.细化.修改而写出来的,在这边感谢那些慷慨提供科普文档的技术大拿们. 要查阅JQ的源文件请下载开发版的JQ.j ...

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

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

  4. jquery源码解析:jQuery工具方法Callbacks详解

    我们首先来讲下Callbacks是如何使用的:第一个例子 function a(){} function b(){} var cb = $.Callbacks(); cb.add(a); cb.add ...

  5. jQuery源码解析资源便签

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

  6. jquery 源码解析

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

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

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

  8. 一.jQuery源码解析之总体架构

    (function (window, undefined) { //构建jQuery对象 var document = window.document, navigator = window.navi ...

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

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

随机推荐

  1. Linux 任务计划

    一.配置cron实现自动化 使用cron实现任务自动化可以通过修改/etc/crontab文件以及使用crontab命令实现,其结果是一样的. 1./etc/crontab文件实现自动化 cron守护 ...

  2. JAVA 多线程和并发学习笔记(四)

    1. 多进程 实现并发最直接的方式是在操作系统级别使用进程,进程是运行在它自己的地址空间内的自包容的程序.多任务操作系统可以通过周期性地将CPU从一个进程切换到另一个进程,来实现同时运行多个进程. 尽 ...

  3. javaEE中关于dao层和services层的理解

    javaEE中关于dao层和services层的理解 入职已经一个多月了,作为刚毕业的新人,除了熟悉公司的项目,学习公司的框架,了解项目的一些业务逻辑之外,也就在没学到什么:因为刚入职, 带我的那个师 ...

  4. TCP/IP之蓟辽督师

    真正的知识是深入浅出的,码农翻身" 公共号将苦涩难懂的计算机知识,用形象有趣的生活中实例呈现给我们,让我们更好地理解. 本文源地址:TCP/IP之蓟辽督师 本文续<TCP/IP之大明内 ...

  5. windows 下 gvim/vim lua支持问题,neocomplete等插件支持

    此文是按照知乎 https://www.zhihu.com/question/29333426 "windows下vim的lua支持问题?" 一文汇总的解决方案. 题主提供了不错的 ...

  6. bash profile .bashrc

    /etc/profile:此文件为系统的每个用户设置环境信息,当用户第一次登录时,该文件被执行. 并从/etc/profile.d目录的配置文件中搜集shell的设置. /etc/bashrc:为每一 ...

  7. POI 设置

    FileOutputStream fos = new FileOutputStream("D:\\15.xls"); HSSFWorkbook wb = new HSSFWorkb ...

  8. HBase的伪分布式安装(原创)

    准备工作: 1)安装了伪分布式hadoop:参照http://blog.csdn.net/zolalad/article/details/11472207 2)修改已安装好的hadoop配置文件: a ...

  9. python-etcd

    Client 对象 ['_MDELETE', '_MGET', '_MPOST', '_MPUT', '__class__', '__contains__', '__del__', '__delatt ...

  10. socket关联查询

    反向代理服务器面对用户请求和后端服务器响应,需要将两者联系起来,于是就涉及到socket关联查询了. 代理服务器处理用户的请求和后端服务器的响应,并且这两方的请求和响应是相互对应的,因此对于代理接收到 ...