Sizzle之tokenize

在Sizzle里,大体思路,当为复合选择器时,判断是否支持querySeletorAll,如果不支持则调用自写方法select。

select的功能十分冗长,下面先分析tokenize

在tokenize函数的作用是将形如'ul.topnav > li,div'的选择器解析为

[

    [{value:'ul',type:"TAG",matches:['ul']},{value:'.topnav',type:"CLASS",matches:['topnav']},{value:' > ',type:">"},{value:'li',type:"TAG",matches:['li']}],

    [{value:'div',type:"TAG",matches:['div']}]

]

每个逗号解析为一个数组,表示一个独立选择器项,每个独立选择器项的选择关系按照CLASS,TAG,ID和关系解析成如上格式。

这种解析的类型是下面进行查找元素的固定格式,先不考虑后面,我们先分析tokenize的实现。

//缓存函数创建函数

var createCache = function(){

    //缓存函数共用数组

    var keys = [];

    return function cache(name,value){

        // 当长度大于某常数时,清除老元素,插入新元素

        if( keys.push( name ) > 1){

            delete cache[ keys.shift() ]

        };

        return cache[name] = value;

    };

}

//缓存函数对象

//这是一个方法,通过这个方法可以将缓存数据保存在这个函数(也是对象)里。

var tokenCache = createCache();

//简易正则过滤对象

var filter = {

    TAG: /^(\w+)/,

    ID: /^#(\w+)/,

    CLASS: /^\.(\w+)/

}

//关系匹配正则

var relation = /^\s*([\>\+\~]|^\s)\s*/;

//逗号正则

var comma = /^\,/;

function tokenize(selector){

    var cached = tokenCache[selector];

    //缓存中有结果,直接返回

    if(cached){

        return cached;

    }

    // 循环条件

    var sofar = selector,

    // 结果数组

    groups = [],

    // 匹配参数

    matched,

    // 一个独立选择器项

    token = [],

    //辅助参数正则匹配结果

    match;

    //循环解析选择器

    while(sofar){

        //首次默认创建一个单独选择器项

        //之后通过判断逗号创建

        if(!matched || (match = comma.exec(sofar)) ){

            if(match){

                //保存未解析的选择器

                sofar = sofar.slice(match[0].length);

            }

            groups.push(token = []);

        }

        //每次循环设置辅助匹配参数为false,如果之后仍无匹配不存在通过break跳出循环

        matched = false;

        //匹配关系>等

        //解析保存结果

        if(match = relation.exec(sofar)){

            matched = match.shift();

            token.push({

                value: matched,

                type: match[0].replace( /\s+/g, " " )

            })

            sofar = sofar.slice(matched.length)

        }

        //匹配选择器CLASS,TAG,ID

        //解析保存结果

        for(var type in filter){

            if(match = filter[type].exec(sofar)){

                matched = match.shift();

                token.push({

                    value: matched,

                    type: type,

                    matches: match

                })

                sofar = sofar.slice(matched.length)

            }

        }

        //没有跳出循环。

        if ( !matched ) {

            break;

        }

    

    }

    //缓存结果并返回

    return (tokenCache(selector,groups));

}

console.log(tokenize('ul.topnav > li,div'))

console.log(tokenize('a li'))

这是我模拟输入输出结果写出的函数,jQuery源码中在实现主要功能的基础上进行了大量兼容处理。

缓存在jQuery里面非常常用,在这里重新说一遍基本思路是,将函数输入值key,和输出值保存于缓存对象中,等到下次调用函数,如果缓存对象中有值,则直接读取。

本例的缓存略微复杂,它是利用函数将缓存数据保存于同名对象中。又,作者希望几种缓存共用长度,在这里用了创建缓存函数的函数。

你可能感兴趣的:(token)