前言

最近看了两遍jQuery源码,感觉只是看懂了jQuery的小部分小部分,不过仅此,就已经对john resig佩服的五体投地咯。。

下面附上这位帅哥的靓照,记住吧,是他改变了世界。


看的大多是实现的细节、技巧,整体的架构还是没有看出来。功力不够呀,还得慢慢修炼。jQuery真的是值得去读无数次的东西,每读一次,都会有不一样的收获。以后有空就多看看吧~

研读优秀框架的源码,是从初级jser进阶到中级jser的捷径,可以学到好多优秀代码的风格。从代码可以看出一个程序员的水平呀~

现在把自己看出来的一些帅呆了的技巧,细节记录下来,一起分享。

jQuery的细节技巧

巧用条件运算符

  1.     trim: trim ?
  2.             function( text ) {
  3.                 return text == null ?
  4.                     "" :
  5.                     trim.call( text );
  6.             } :
  7.             function( text ) {
  8.                 return text == null ?
  9.                     "" :
  10.                     text.toString().replace( trimLeft, "" ).replace( trimRight, "" );
  11.        

这里用了一个条件运算符,如果String.trim存在的话就用,不存在就用自己实现的trim函数。在jQuery的源码中,用了大量的这种技巧,比如:

  1. data = data === "true" ? true :
  2.     data === "false" ? false :
  3.     data === "null" ? null :
  4.     !jQuery.isNaN( data ) ? parseFloat( data ) :
  5.         rbrace.test( data ) ? jQuery.parseJSON( data ) : data;
  6. for ( var i = one ? index : 0, max = one ? index + 1 : options.length; i < max; i++ ) {} 

多用点这种技巧,就不用满屏幕都是if & else 了。代码会变得简洁,轻巧。

jQuery的循环遍历技巧

下面列出一些自己看出的、常见的、平时代码中常用的循环技巧

1、

  1.     
  2.     // 简单的for-in(事件) 
  3.     for ( type in events ) { 
  4.  
  5.     }

2、

  1.     // 缓存length属性,避免每次都去查找length属性,稍微提升遍历速度 
  2.     for ( var j = 0, l = arrow.length; j < l; j++ ) { 
  3.  
  4.    

3、

  1.     // 不比较下标,直接判断元素是否为true(强制类型转换) 
  2.     var elem; 
  3.     for ( var i = 0; elems[i]; i++ ) { 
  4.         elem = elems[i]; 
  5.         // ... 
  6.     }

4、

  1.     
  2.     // 遍历动态数组(事件),不能缓存length属性,j++之前先执行j--
  3.     for ( j = 0; j < eventType.length; j++ ) { 
  4.         eventType.splice( j--, 1 ); 
  5.     } 
  6.     for ( var i = 1; i < results.length; i++ ) { 
  7.         if ( results[i] === results[ i - 1 ] ) { 
  8.             results.splice( i--, 1 ); 
  9.         } 
  10.     }

这个细节要注意哈,稍不注意就会在这里折腾很久。动态数组不能缓存!

5、

  1.     // 迭代过程中尽可能减少遍历次数(事件),从pos开始
  2.     for ( j = pos || 0; j < eventType.length; j++ ) { 
  3.  
  4.     }

6、

  1.     //倒序遍历,有利于浏览器优化,稍微提升遍历速度 
  2.     for ( var i = this.props.length, prop; i; ) { 
  3.         prop = this.props[ --]; 
  4.         event[ prop ] = originalEvent[ prop ]; 
  5.     }

7、

  1.     
  2.     // while检查下标i 
  3.     var i = arr.length; 
  4.     while( i-- ) { 
  5.         obj[ arr[i] ] = deferred[ arr[i] ]; 
  6.     }

通过闭包,返回前一作用域的参数

  1.     // 闭包。返回的函数保持对guid的引用。
  2.     function a(){
  3.         var guid = 1;
  4.         return function(){
  5.             return guid++;
  6.         }
  7.     }
  8.     var defer = a();

一次实现多种方法

  1.     if ( rfxtypes.test(val) ) {
  2.         // 如果是toggle,则判断当前是否hidden,如果hidden则show,否则hide
  3.         // 如果不是toggle,说明val是hide/show之一
  4.         val === "toggle" ? (hidden ? "show" : "hide") : val ;
  5.  
  6.     } else { ... }

条件运算符 + 关联数组 一次实现三种方法,帅呆了!不过貌似这种方法平时用的不多。

同样的思想还有一处:

  1.     jQuery.each(["live", "die"], function( i, name ) {
  2.         jQuery.fn[ name ] = function( types, data, fn, origSelector /* Internal Use Only */ ) {
  3.             if ( name === "die" && !types &&
  4.                         origSelector && origSelector.charAt(0) === "." ) {
  5.                 context.unbind( origSelector );
  6.                 return this;
  7.             }
  8.             //....
  9.             return this;
  10.         };
  11.     }); 

一次性实现两个方法:

  1.     jQuery.each( ("blur focus focusin focusout load resize scroll unload click dblclick " +
  2.         "mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave " +
  3.         "change select submit keydown keypress keyup error").split(" "), function( i, name ) {
  4.         // Handle event binding
  5.         jQuery.fn[ name ] = function( data, fn ) {
  6.            return arguments.length > 0 ?
  7.                this.bind( name, data, fn ) :
  8.                this.trigger( name );
  9.         };
  10.         // 将事件名注册(添加)到jQuery.attrFn,当遇到对同名属性的操作时,转换为对事件接口的调用
  11.         if ( jQuery.attrFn ) {
  12.            jQuery.attrFn[ name ] = true;
  13.         }
  14.    });

通过each方法,实现n多方法。到此,除了惊叹还是惊叹。

实例化jQuery对象

先看代码:

  1.     jQuery = function( selector, context ) {
  2.         return new jQuery.fn.init( selector, context, rootjQuery );
  3.     }

这里jQuery对象就是jQuery.fn.init对象

如果执行new jQeury(),生成的jQuery对象会被抛弃,最后返回 jQuery.fn.init对象;

因此可以直接调用jQuery( selector, context ),没有必要使用new关键字
还有一行代码如下:

  1.     jQuery.fn.init.prototype = jQuery.fn = jQuery.prototype

所有挂载到jQuery.fn的方法,相当于挂载到了jQuery.prototype,即挂载到了jQuery 函数上(一开始的 jQuery = function( selector, context ) ),但是最后都相当于挂载到了 jQuery.fn.init.prototype,即相当于挂载到了一开始的jQuery 函数返回的对象上,即挂载到了我们最终使用的jQuery对象上。

这里的init函数里面执行了,我们平时接触具多的选择器。简单分析如下:

  1.     
  2.     //如果传入一个DOM元素。。包装成jQuery对象,直接返回。
  3.     if ( selector.nodeType ) {
  4.         this.context = this[0] = selector;
  5.         this.length = 1;
  6.         return this;
  7.     }
  8.     //如果传入一个字符串
  9.         //如果字符串时一个类似"<a>hello</a>"样子的,直接生成。
  10.         //传入ID,直接document.getElementById(),然后包装成jQuery对象。
  11.     //选择器表达式,find...
  12.     //如果传入函数,则为ready开始 

jQuery 巧妙避免ReferenceError

在源码中,经常看到这样的代码:

args = args || [];

这样可以确保args可用,一个避免null,underfined造成的ReferenceError的常见技巧。

比如我们还可以看到这样的:

  1.     isArray: Array.isArray || function( obj ) {
  2.         return jQuery.type(obj) === "array";
  3.     }//这里和上面的条件运算符效果差不多。
  4.  
  5.     name = jQuery.cssProps[ origName ] || origName;

在很多需要传入选项对象的函数中,如果用户自己传入,就用用户传进的;如果用户没有传值,则用开发者默认的参数。这样的代码很帅气有木有!

jQuery的实现框架

  1.     (function( window, undefined ) {
  2.  
  3.         // 构造jQuery对象
  4.         var jQuery = function( selector, context ) {
  5.                 return new jQuery.fn.init( selector, context, rootjQuery );
  6.             }
  7.         // 工具函数 Utilities
  8.         // 异步队列 Deferred
  9.         // 浏览器测试 Support
  10.         // 数据缓存 Data
  11.         // 队列 queue
  12.         // 属性操作 Attribute
  13.         // 事件处理 Event
  14.         // 选择器 Sizzle
  15.         // DOM遍历
  16.         // DOM操作
  17.         // CSS操作
  18.         // 异步请求 Ajax
  19.         // 动画 FX
  20.         // 坐标和大小
  21.         window.jQuery = window.= jQuery;
  22.  
  23.  })(window); 

结语

以上分析,都只是jQuery的一些皮毛,还有很多很多没有发现,以后有时间,有机会再研究咯~

总之每看一次,都会有很大的收获,成长。

最近状态不好,做事效率很低。以后看书码代码的时候,要关了qq,微博,河畔。。。
离开用心,新的开始。虽然很多不舍,但必须面对,做出选择。对于有些人,应该就是永远不见,有些人,真舍不得。

做喜欢的事,做好喜欢的事。

从jQuery学细节的更多相关文章

  1. jQuery基础细节

    使用CDN版的jQuery         用托管在公共CDN(Content Delivery Network,内容分发网络)上的jQuery库,CDN是一种内容分发网络,当用户请求其数据时,CDN ...

  2. jquery冲突细节

    使用jquery报一个错误,之前也遇到过,今天记录下来,方便以后使用 Uncaught TypeError: Object #<Object> has no method 'test' 这 ...

  3. JQuery选择器细节-遁地龙卷风

    1.层次选择器-子元素选择器 <body> <div> <p>lol</p> <div> <p></p> </ ...

  4. 【深入浅出jQuery】源码浅析--整体架构

    最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...

  5. jQuery整体架构源码解析(转载)

    jQuery整体架构源码解析 最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性, ...

  6. jQuery整体架构源码解析

    最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...

  7. 【深入浅出jQuery】源码浅析--整体架构(转)

    最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,让你感叹代码之美. 其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐 ...

  8. jquery源码解读

    最近一直在研读 jQuery 源码,初看源码一头雾水毫无头绪,真正静下心来细看写的真是精妙,其结构明晰,高内聚.低耦合,兼具优秀的性能与便利的扩展性,在浏览器的兼容性(功能缺陷.渐进增强)优雅的处理能 ...

  9. JavaScript和jQuery的学习

    还有12天就要回学校了,我的假期计划还能实现吗?在这12天里,需要把JavaScript和jQuery学完.我知道这两个技术对于前端网页开发非常重要.前期把HTML和CSS学完了,学的不是特别深,只是 ...

随机推荐

  1. java.lang.NoClassDefFoundError: org/springframework/expression/PropertyAccessor

    这个异常原因种类不一,网上有各个版本,本人的是因为缺少了spring-expression-3.2.1.RELEASE.jar 2015-9-18 23:19:11 org.apache.catali ...

  2. Spring笔记:AOP基础

    Spring笔记:AOP基础 AOP 引入AOP 面向对象的开发过程中,我们对软件开发进行抽象.分割成各个模块或对象.例如,我们对API抽象成三个模块,Controller.Service.Comma ...

  3. 011_用eclipse打开hadoop1.2.1源码出现The method getSplits(JobConf, int) is undefined for the type Object错误

    出现的错误截图: 此时Eclipse使用的jdk1.8,将编译环境改成jdk1.7就行了,解决. 没问题了. 下面观点是参考如下网址,未经验证.参考:http://blog.csdn.net/onei ...

  4. mysql安装配置 (单个mysql安装)

    mysql安装于 c盘因为有注册文件 和注册表 1.为了防止mysql数据表变大和sql日志增加   改变C:\ProgramData\MySQL\MySQL Server 5.7\my.ini 得指 ...

  5. Linux用户和用户组管理 用户配置和管理的相关文件

    用户信息文件 /etc/passwd 这个文件中保存的就是系统中所有的用户及其对应的用户主要信息. 文件格式 :  第1字段 第2字段 第3字段 第4字段 第5字段 第6字段 第7字段 用户名称 密码 ...

  6. JavaScript笔记04——事件与回调

    1.在浏览器中,大多数代码都是由事件驱动的(event-driven). 这和生物中的神经反射有点类似. 比如说,谷歌页面上的一个按钮, 当我们“按下”这个按钮的时候,将跳出如下界面. 那么你有没想过 ...

  7. 机器学习性能指标之ROC和AUC理解与曲线绘制

    一. ROC曲线 1.roc曲线:接收者操作特征(receiveroperating characteristic),roc曲线上每个点反映着对同一信号刺激的感受性. 横轴:负正类率(false po ...

  8. shell脚本:批量修改文件名

    参考链接1:shell脚本:批量修改文件名(删除文件名中字符) 参考链接2:linux shell 字符串操作详解 (长度,读取,替换,截取,连接,对比,删除,位置 ) 参考链接3:每天一个linux ...

  9. HDU 5183 Negative and Positive (NP) (hashmap+YY)

    学到了以邻接表方式建立的hashmap 题意:给你一串数a和一个数k,都有正有负,问知否能找到一对数(i,j)(i<=j)保证a [i] - a [i+1] + a [i+2] - a [i+3 ...

  10. 【转】Vue.js:轻量高效的前端组件化方案

    摘要:Vue.js通过简洁的API提供高效的数据绑定和灵活的组件系统.在前端纷繁复杂的生态中,Vue.js有幸受到一定程度的关注,目前在GitHub上已经有5000+的star.本文将从各方面对Vue ...