jquery ready函数源代码研究_jquery

一般情况下都是设置body标签的onload监听window的load事件.但load事件是要在页面的元素全部加载完了才触发的,如果页面上图片较多或图片太大,就会导致初始化的代码未被执行的时候用户就做了其它操作了. Jquery库提供了一个非常方便好用的函数( $(selector).ready()),让我们可以在页面的dom加载完后就可以做相应的操作(当然,这还得看用户浏览器的支持).,而不用等待全部元素加载完成.例如:
$(document).ready(function (){ alert('use in page script tag') });
$(document).ready(function (){ alert('use in import js file') });
现在让我们来研究一下这个函数的实现.
原理:
在jquery脚本加载的时候,会设置一个isReady的标记,监听DOMContentLoaded事件(这个不是什么浏览器都有的,不同浏览器,jquery运作方式不一样).当然遇到调用ready函数的时候,如果isReady未被设置,那就是说页面未加载完,就会把要执行的函数用一个数组缓存起来,当页面加载完后,再把缓存的函数一一执行.
Jquery中的详细代码分析:

复制代码 代码如下:

ready: function(fn) {
        // 绑定监听器
        bindReady();
        // 如果 DOM 加载完成
        if ( jQuery.isReady )
            // 马上运行此函数
            fn.call( document, jQuery );
        // 否则保存起来
        else
            // 把函数加入缓存数组中
            jQuery.readyList.push( function() { return fn.call(this, jQuery); } );
        return this;
}

让我们看看jquery如果实现不同浏览器dom加载完成的通知 bindReady()函数:

复制代码 代码如下:

var readyBound = false;
function bindReady(){
    if ( readyBound ) return;
    readyBound = true;

// Mozilla,opera,webkitnightlies支持DOMContentLoaded事件
    if ( document.addEventListener && !jQuery.browser.opera)
        // 直接使用事件回调即可
        document.addEventListener( "DOMContentLoaded", jQuery.ready, false );

    // 如果是ie并且不是嵌在frame中
    // 就需要不断地检查文档是否加载完
    if ( jQuery.browser.msie && window == top ) (function(){
        if (jQuery.isReady) return;
        try {
            // 这个地方标记一下,在后面解析(1)
            document.documentElement.doScroll("left");
        } catch( error ) {
//// 这个地方标记一下,在后面解析(2)
            setTimeout( arguments.callee, 0 );
            return;
        }
        // and execute any waiting functions
        jQuery.ready();
    })();

    if ( jQuery.browser.opera )
        document.addEventListener( "DOMContentLoaded", function () {
            if (jQuery.isReady) return;
            for (var i = 0; i < document.styleSheets.length; i++) // 标记(3)
                if (document.styleSheets[i].disabled) {
                    setTimeout( arguments.callee, 0 );
                    return;
                }
            // and execute any waiting functions
            jQuery.ready();
        }, false);

    if ( jQuery.browser.safari ) {
        var numStyles;
        (function(){
            if (jQuery.isReady) return;
            if ( document.readyState != "loaded" && document.readyState != "complete" ) { // 标记(4)
                setTimeout( arguments.callee, 0 );
                return;
            }
            if ( numStyles === undefined )
                numStyles = jQuery("style, link[rel=stylesheet]").length;
            if ( document.styleSheets.length != numStyles ) { // 标记(5)
                setTimeout( arguments.callee, 0 );
                return;
            }
            // and execute any waiting functions
            jQuery.ready();
        })();
    }

    // A fallback to window.onload, that will always work
    jQuery.event.add( window, "load", jQuery.ready ); // 标记(6)
}
}

(1):这个主要是测出ie下的dom ready,原理在这里http://javascript.nwbox.com/IEContentLoaded/,利用在ie下.当dom未完成解析时,调用document的document.documentElement.doScroll(”left”)会出错这个小技巧便可得知dom有没有ready了.
(2):setTimeout( arguments.callee, 0 )这句是表示延迟0秒调用,实际上它不会马上就调用,而是会尽可能快地调用,它告诉浏览器为当前任何挂起的事件运行完事件句柄并且完成了文档当前状态的更新后才调用. Arguments.callee即是外层的匿名函数,参数的调用者
(3):这个地方你也许觉得奇怪,为什么不在mozilla那里一起处理呢? 原因就是opera的DOMContentLoaded事件发生后,其css样式是还没完全可用的,所以要特殊处理,就是判断每个css的tag都是不是enable了.
(4),(5):safari中document.readyState的状态为loaded或complete时,css文件引入还未能确定是不是解析完了的,所以要通过判断其css文件数目
(6):最后,如果上面的hack都不支持的话…就用最保险的load事件,保证能执行到初始化代码.

时间: 2024-08-27 15:04:09

jquery ready函数源代码研究_jquery的相关文章

jQuery ready函数滥用分析_jquery

查看下之前写的代码,有许多这样的代码: 复制代码 代码如下: $(function(){ //do something. }); 这里面可能写一些初始化样式.给其他dom元素附加事件处理等.刚开始没发现什么问题,但是在页面引用了一些下载缓慢的图片时,问题出现了:在页面html下载完,到所有资源全部下载完之间,绑定dom元素上绑定的事件无法执行.用脚本绑定的样式无效等混乱情况,也就是ready么有执行. Ok,我的情况比较特殊,可能这个情况对于大部分同学是不可能遇到的,但是在开发中我们不得不考虑在

jquery ready函数深入分析_C 语言

最近看一些关于jquery ready 有人说他缓慢,有人说他快,说法不一. 于是自己深入研究一下.首先看了一下jquery 文档 关于ready 的描述 While JavaScript provides the load event for executing code when a page is rendered, this event does not get triggered until all assets such as images have been completely

jQuery实用函数用法总结_jquery

本文以实例的形式总结了jQuery的常见实用函数.分享给大家供大家参考之用.具体示例如下: 1.修剪字符串 $('#id').val($.trim($('#someid').val()))   2.遍历集合 可能这样写: var anArray = ['one','two']; for(var n = 0; n < anArray.length; n++){ } 还有可能这样写: var anObject = {one: 1, two: 2}; for(var p in anObject){ }

一个超简单的jQuery回调函数例子(分享)_jquery

jQuery回调函数简单使用 比如说,我们想要点击某个按钮后触发事件, 先把一些指定内容给隐藏掉, 然后跳出相关信息的对话框. 如果使用普通的方法, 不用回调函数的话, 会有怎么样的效果呢? 效果是先弹出对话框再隐藏内容, 然后再隐藏指定内容. 这显然不是我们想要的效果, 如果使用回调函数,就可以解决这个问题. 当然,回调函数功能远不只这么简单-- 具体的代码如下: <%@ page language="java" import="java.util.*" p

jQuery 自定义函数写法分享_jquery

自定义主要通过两种方式实现$.extend({aa:function(){}});$.fn.extend({aa:function(){}});调用的方法分别是:$.aa();$($this).aa(); 注意:创建函数时不要放在 $(function() { }中,调用时候要放在事件里面$($this).click(function(){$.aa();}); jQuery.extend 函数详解JQuery的extend扩展方法:Jquery的扩展方法extend是我们在写插件的过程中常用的方

jquery队列函数用法实例_jquery

本文实例讲述了jquery队列函数用法.分享给大家供大家参考.具体实现方法如下: 复制代码 代码如下: <!DOCTYPE> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <script type="text/javascript" src="jquery-1.9

jQuery 工具函数学习资料_jquery

URL 字符串操作 数组和对象操作 测试操作 浏览器 1:URL操作: $.param(obj) 返回 :string: 说明:将jquery对象按照name/value 或者key/value序列化为URL参数,用&连接. 示例: var obj ={name:zh,age:20}; alert(jQuery.param(obj)); //alert "name=zh&age=20";   2:字符串操作: jQuery.trim(str) 返回:string: 说明

jquery ready函数、css函数及text()使用示例_jquery

复制代码 代码如下: <script type="text/javascript" src="/javascript/jquery.js"></script> <script language="javascript"> $("#CheckoutStepShippingProvider").ready(function(){ if($("#Note_52413376ea55e_1&

jQuery setTimeout()函数使用方法_jquery

setTimeout()从载入后延迟指定的时间去执行一个表达式或者是函数;仅执行一次 ;和window.clearTimeout一起使用. 我在 复制代码 代码如下: $(document).ready(function(){setTimout(test(),200); function test() { alert(1); } }); 只会执行一次,有朋友说可以使用 复制代码 代码如下: setInterval ("showTime()", 5000);function showTi