17
Jan 2014
source code selector
jQuery

学习jQuery源码-选择器1——从init函数开始的初步分析

字数:7173

作者:Jerry

感谢Aaron的jQuery源码分析系列

本系列文章是学习jQuery源码的笔记,基于网络的各种教程与自己的理解而写(不会覆盖到每个点,可能存在错误),一来是记录学习成果;),以后要用到时能方便找到;二来也帮助一下需要的人。

jQuery选择器入门

选择器,是jQuery的基石。

我相信,jQuery如此流行的最重要原因就是它强大而便捷的选择器。全面的css/css3语法支持,准确定位到每一个元素,jQuery极大地提高了获取、筛选元素的效率。

从正则rquickExpr说起

说高效强大莫如正则,jQuery中运用了大量的正则来判断、切分字符串。现在说一说选择器中用到的rquickExpr。

rquickExpr = /^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,

分解一下:

  1. |之前:^(?:\s*(<[\w\W]+>)[^>]*

    • \s* : 匹配任何空白字符,包括空格、制表符、换页符等等 零次或多次
    • [\w\W]+ : 匹配'[A-Za-z0-9]'或[^A-Za-z0-9]' 一次或多次
    • [^>]:除了>的任意字符 零次或多次

    这半截是匹配 <p> 这样的,前面可加任意空白符,<>内可以是任意字符,后面可以是任意字符。

  2. |之后:#([\w-]*))$/

    匹配 以#号开始,结尾是[A-Za-z0-9_-]出现任意次,如 #idExpress

例子:

rquickExpr.exec(' < > hi');  
// 输出  [" < > hi", "< >", undefined]

rquickExpr.exec(' <p>text</p>');  
// 输出  [" <p>text</p>", "<p>text</p>", undefined]

rquickExpr.exec('#hi-hello');  
// 输出  ["#hi-hello", undefined, "hi-hello"]

rquickExpr.exec('#    hi-hello');  
// 输出  null

综合一下,rquickExpr就是匹配html标记(<前可有空白,>后可以有任意字符)或者"#id"形式的id表达式(前后不可空白)。

jQuery.fn.init函数分析

jQuery.fn.init其实是选择器的第一道接口,本身处理一些简单的选择器,更复杂的转交Sizzle来处理。

// jQuery.fn.init函数

init : function( selector, context, rootjQuery ) {

    var match, elem;

    // HANDLE: $(""), $(null), $(undefined), $(false)
    if ( !selector ) {
        return this;
    }

    // Handle HTML strings
    // 处理字符串 
    if ( typeof selector === "string" ) {
        //是 "<"开始,">"结尾 ,长度>=3,假设就是html标签,不用正则检查,直接match = [ null, selector, null ];
        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 {//否则rquickExpr.exec
            match = rquickExpr.exec( selector );//必须是<htmltag>或者#id两种形式之一
        }

        // Match html or make sure no context is specified for #id
        // 匹配的html(<tag>)  或  确保没有上下文指定为 #id
        if ( match && (match[1] || !context) ) {

            // HANDLE: $(html) -> $(array)
            // match[1](html标记)存在,处理$(html) -> $(array),也就是处理的是html方式
            if ( match[1] ) {
                context = context instanceof jQuery ? context[0] : context;

                // scripts is true for back-compat
                // parseHTML,根据match[1]字符串转换为一组DOM元素
                jQuery.merge( this, jQuery.parseHTML(
                    match[1],
                    context && context.nodeType ? context.ownerDocument || context : document,
                    true
                ) );

                // HANDLE: $(html, props)
                // rsingleTag匹配一个独立的标签,没有任何属性和子节点的字符串;比如'<html></html>'或者'<div></div>'这样
                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) 处理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 ) {
                    // Inject the element directly into the jQuery object
                    this.length = 1;
                    this[0] = elem;
                }

                this.context = document;
                this.selector = selector;
                return this;
            }

        // HANDLE: $(expr, $(...))
        // <htmltag>或者#id情况在上面已被处理,复杂的selector表达式(如div > input#name)之类的在下面处理
        } 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) 处理dom元素
    } else if ( selector.nodeType ) {
        this.context = this[0] = selector;
        this.length = 1;
        return this;

    // HANDLE: $(function)
    // Shortcut for document ready   选择器为函数,则是jQuery.ready的快捷写法
    } else if ( jQuery.isFunction( selector ) ) {
        return rootjQuery.ready( selector );
    }

    // selector是jQuery对象
    if ( selector.selector !== undefined ) {
        this.selector = selector.selector;
        this.context = selector.context;
    }

    return jQuery.makeArray( selector, this );
}


分析:

结束

本篇是选择器的初步介绍,或者说接触到了选择器,其实都是一些简单的selector处理,处理复杂的selector的Sizzle只是露了个面。下一章开始介绍Sizzle。