zoukankan      html  css  js  c++  java
  • jQuery1.11源码分析(9)-----初始化jQuery对象的函数和关联节点获取函数

    这篇也没什么好说的,初始化jQuery对象的函数要处理多种情况,已经被寒冬吐槽烂了。关联节点获取函数主要基于两个工具函数dir和sibling,前者基于指定的方向遍历,后者则遍历兄弟节点(真的不能合并?)。后面的一些API则主要调用这两个函数。大几百行代码,不过逻辑很简单

    // Initialize a jQuery object
    // A central reference to the root jQuery(document)
    var rootjQuery,
    	// Use the correct document accordingly with window argument (sandbox)
    	document = window.document,
    
    	// A simple way to check for HTML strings
    	// Prioritize #id over  to avoid XSS via location.hash (#9521)
    	// Strict HTML recognition (#11290: must start with <)
        //一个检查字符串是否包含HTML的简单正则。
        //需要避免如<script>alert(1)</script>之类的XSS
    	rquickExpr = /^(?:s*(<[wW]+>)[^>]*|#([w-]*))$/,
        //接下来就是被寒冬黑得不亦乐乎的真正的产生jQuery对象的函数了
    	init = jQuery.fn.init = function( selector, context ) {
    		var match, elem;
    
    		// HANDLE: $(""), $(null), $(undefined), $(false)
    		if ( !selector ) {
                //??????为什么这里return 的是一个空数组?
                //这只是显示问题
                console.log(this.length);
    			return this;
    		}
    
    		// Handle HTML strings
            //处理HTML字符串
            //当传入的是字符串时
    		if ( typeof selector === "string" ) {
                //当传入的字符串类似于"<div >"
    			if ( selector.charAt(0) === "<" && selector.charAt( selector.length - 1 ) === ">" && selector.length >= 3 ) {
    				// Assume that strings that start and end with <> are HTML and skip the regex check
    				match = [ null, selector, null ];
    
    			} else {
    				match = rquickExpr.exec( selector );
    			}
    
    			// Match html or make sure no context is specified for #id
    			if ( match && (match[1] || !context) ) {
    
    				// HANDLE: $(html) -> $(array)
    				if ( match[1] ) {
    					context = context instanceof jQuery ? context[0] : context;
    
    					// scripts is true for back-compat
    					// Intentionally let the error be thrown if parseHTML is not present
                        //把DOM元素加到this里
    					jQuery.merge( this, jQuery.parseHTML(
    						match[1],
    						context && context.nodeType ? context.ownerDocument || context : document,
    						true
    					) );
    
    					// HANDLE: $(html, props)
                        //什么情况下context是PlainObject?
                        //处理$(html,props)这种情况。。所以第二个参数是{attrName:attrValue},context此时为PlainObject
    					if ( rsingleTag.test( match[1] ) && jQuery.isPlainObject( context ) ) {
    						for ( match in context ) {
    							// Properties of context are called as methods if possible
                                //为什么这里要调用函数?
    							if ( jQuery.isFunction( this[ match ] ) ) {
    								this[ match ]( context[ match ] );
    
    							// ...and otherwise set as attributes
    							} else {
    								this.attr( match, context[ match ] );
    							}
    						}
    					}
    
    					return this;
    
    				// HANDLE: $(#id)
    				} else {
    
                        //先尝试使用原生借口
    					elem = document.getElementById( match[2] );
    
    					// Check parentNode to catch when Blackberry 4.6 returns
    					// nodes that are no longer in the document #6963
    					if ( elem && elem.parentNode ) {
    						// Handle the case where IE and Opera return items
    						// by name instead of ID
    						if ( elem.id !== match[2] ) {
    							return rootjQuery.find( selector );
    						}
    
    						// Otherwise, we inject the element directly into the jQuery object
                            //这里不能用push,是因为this是一个伪数组,但不能用自己写的push?
    						this.length = 1;
    						this[0] = elem;
    					}
    
    					this.context = document;
    					this.selector = selector;
    					return this;
    				}
    
    			// HANDLE: $(expr, $(...))
    			} else if ( !context || context.jquery ) {
    				return ( context || rootjQuery ).find( selector );
    
    			// HANDLE: $(expr, context)
    			// (which is just equivalent to: $(context).find(expr)
                //这里就是转换调用一下,这种思路可以学习
    			} else {
    				return this.constructor( context ).find( selector );
    			}
    
    		// HANDLE: $(DOMElement)
    		} else if ( selector.nodeType ) {
    			this.context = this[0] = selector;
    			this.length = 1;
    			return this;
    
    		// HANDLE: $(function)
    		// Shortcut for document ready
    		} else if ( jQuery.isFunction( selector ) ) {
    			return typeof rootjQuery.ready !== "undefined" ?
    				rootjQuery.ready( selector ) :
    				// Execute immediately if ready is not present
                    //否则立刻执行
    				selector( jQuery );
    		}
            //如果传进来的是jQuery对象
    		if ( selector.selector !== undefined ) {
    			this.selector = selector.selector;
    			this.context = selector.context;
    		}
    
    		return jQuery.makeArray( selector, this );
    	};
    
    // Give the init function the jQuery prototype for later instantiation
    //将init这个函数的原型设置为jQuery.fn,这样每个jQuery对象都可以共享jQuery.fn上的函数
    init.prototype = jQuery.fn;
    
    // Initialize central reference
    //文档节点的jQuery对象
    rootjQuery = jQuery( document );
    
    //后面关联节点时用来判断函数名是否带有Until或All的正则
    var rparentsprev = /^(?:parents|prev(?:Until|All))/,
    	// methods guaranteed to produce a unique set when starting from a unique set
        //标记某些方法是否需要确保返回的集合里每一个元素都唯一
    	guaranteedUnique = {
    		children: true,
    		contents: true,
    		next: true,
    		prev: true
    	};
    
    jQuery.extend({
        //按某一方向查找,返回匹配元素数组,注意这里有until
    	dir: function( elem, dir, until ) {
    		var matched = [],
    			cur = elem[ dir ];
    		while ( cur && cur.nodeType !== 9 && (until === undefined || cur.nodeType !== 1 || !jQuery( cur ).is( until )) ) {
    			if ( cur.nodeType === 1 ) {
    				matched.push( cur );
    			}
    			cur = cur[dir];
    		}
    		return matched;
    	},
        //按兄弟元素方向查找,返回匹配元素数组
    	sibling: function( n, elem ) {
    		var r = [];
    
    		for ( ; n; n = n.nextSibling ) {
    			if ( n.nodeType === 1 && n !== elem ) {
    				r.push( n );
    			}
    		}
    
    		return r;
    	}
    });
    
    jQuery.fn.extend({
        //这个函数应该和之前的is待在一块
    	has: function( target ) {
    		var i,
    			targets = jQuery( target, this ),
    			len = targets.length;
    
    		return this.filter(function() {
    			for ( i = 0; i < len; i++ ) {
    				if ( jQuery.contains( this, targets[i] ) ) {
    					return true;
    				}
    			}
    		});
    	},
        //匹配jQuery对象中每一个DOM元素最接近的父元素,最后会去重
    	closest: function( selectors, context ) {
    		var cur,
    			i = 0,
    			l = this.length,
    			matched = [],
    			pos = rneedsContext.test( selectors ) || typeof selectors !== "string" ?
    				jQuery( selectors, context || this.context ) :
    				0;
    
    		for ( ; i < l; i++ ) {
    			for ( cur = this[i]; cur && cur !== context; cur = cur.parentNode ) {
    				// Always skip document fragments
                    //跳过文档碎片节点
    				if ( cur.nodeType < 11 && (pos ?
                        //pos此时是一个jQuery对象
    					pos.index(cur) > -1 :
    
    					// Don't pass non-elements to Sizzle
    					cur.nodeType === 1 &&
                            //这里是检查cur是否和selectors匹配
    						jQuery.find.matchesSelector(cur, selectors)) ) {
                        //因为只匹配最近的一个节点
    					matched.push( cur );
    					break;
    				}
    			}
    		}
    
    		return this.pushStack( matched.length > 1 ? jQuery.unique( matched ) : matched );
    	},
    
    	// Determine the position of an element within
    	// the matched set of elements
        //其实本质上还是调用工具函数inArray,不过inArray设计得不错,错误返回-1,正确返回索引
    	index: function( elem ) {
    
    		// No argument, return index in parent
            //这个获得索引的方法很巧妙,检查自己前面有多少元素,就是自己的索引
    		if ( !elem ) {
    			return ( this[0] && this[0].parentNode ) ? this.first().prevAll().length : -1;
    		}
    
    		// index in selector
            //处理传进来的参数是elem的情况
    		if ( typeof elem === "string" ) {
    			return jQuery.inArray( this[0], jQuery( elem ) );
    		}
    
    		// Locate the position of the desired element
    		return jQuery.inArray(
    			// If it receives a jQuery object, the first element is used
    			elem.jquery ? elem[0] : elem, this );
    	},
        //这个添加方式颇为奇葩。。添加完还要unique一下。。添加完原有的selector也会没了。。
    	add: function( selector, context ) {
    		return this.pushStack(
    			jQuery.unique(
    				jQuery.merge( this.get(), jQuery( selector, context ) )
    			)
    		);
    	},
        //把压栈之前的jQuery对象加过来,估计是后面有用
    	addBack: function( selector ) {
    		return this.add( selector == null ?
    			this.prevObject : this.prevObject.filter(selector)
    		);
    	}
    });
    
    //之所以需要这么个工具函数,是因为在遍历的时候需要考虑无用节点。。不过其实也没用几次。。
    function sibling( cur, dir ) {
    	do {
    		cur = cur[ dir ];
    	} while ( cur && cur.nodeType !== 1 );
    
    	return cur;
    }
    
    //这里为什么是each而不是extend?
    //这里使用each的形式对下面的每一个函数进行处理
    //下面这些函数就是调用前面的API,说明前面抽象得比较好
    jQuery.each({
    	parent: function( elem ) {
    		var parent = elem.parentNode;
    		return parent && parent.nodeType !== 11 ? parent : null;
    	},
    	parents: function( elem ) {
    		return jQuery.dir( elem, "parentNode" );
    	},
    	parentsUntil: function( elem, i, until ) {
    		return jQuery.dir( elem, "parentNode", until );
    	},
    	next: function( elem ) {
    		return sibling( elem, "nextSibling" );
    	},
    	prev: function( elem ) {
    		return sibling( elem, "previousSibling" );
    	},
    	nextAll: function( elem ) {
    		return jQuery.dir( elem, "nextSibling" );
    	},
    	prevAll: function( elem ) {
    		return jQuery.dir( elem, "previousSibling" );
    	},
    	nextUntil: function( elem, i, until ) {
    		return jQuery.dir( elem, "nextSibling", until );
    	},
    	prevUntil: function( elem, i, until ) {
    		return jQuery.dir( elem, "previousSibling", until );
    	},
    	siblings: function( elem ) {
    		return jQuery.sibling( ( elem.parentNode || {} ).firstChild, elem );
    	},
    	children: function( elem ) {
    		return jQuery.sibling( elem.firstChild );
    	},
    	contents: function( elem ) {
            //这里要判断元素是否是iframe
    		return jQuery.nodeName( elem, "iframe" ) ?
    			elem.contentDocument || elem.contentWindow.document :
    			jQuery.merge( [], elem.childNodes );
    	}
        //处理函数,进行一层封装
    }, function( name, fn ) {
    	jQuery.fn[ name ] = function( until, selector ) {
    		var ret = jQuery.map( this, fn, until );
    
    		if ( name.slice( -5 ) !== "Until" ) {
                //当name不为xxxUntil时,说明第一个参数不是用来until的,而是选择符。
    			selector = until;
    		}
            //含有选择符的话要过滤一下
    		if ( selector && typeof selector === "string" ) {
    			ret = jQuery.filter( selector, ret );
    		}
    
    		if ( this.length > 1 ) {
    			// Remove duplicates
                //当需要去重的时候去重
    			if ( !guaranteedUnique[ name ] ) {
    				ret = jQuery.unique( ret );
    			}
    
    			// Reverse order for parents* and prev-derivatives
    			if ( rparentsprev.test( name ) ) {
    				ret = ret.reverse();
    			}
    		}
    
    		return this.pushStack( ret );
    	};
    });
    
  • 相关阅读:
    【转载】VM Role实战: Azure 平台上传虚拟机C 盘不能大于64GB
    Asp.Net MVC项目编译View
    Notepad测试数据库连接
    【转载】使用远程桌面Remote desktop访问Azure虚拟机
    【转载】Azure Blob Storage 云存储中实现大文件分块断点续传
    jQuery Mobile 笔记(1) jQuery Mobile页面结构
    【转载】使用Azure SDK 1.6 ,VM Role里加载Azure Drive方案
    大把大把的时间 你要怎么把握啊
    asp.net 本质论学习
    在客户端用javascript或VBSCRIPT生成WORD文档(转)
  • 原文地址:https://www.cnblogs.com/suprise/p/3662218.html
Copyright © 2011-2022 走看看