亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > JavaScript > 正文

自己動手實現(xiàn)jQuery Callbacks完整功能代碼詳解

2019-11-20 21:35:30
字體:
供稿:網(wǎng)友

用法和$.Callbacks完全一致 , 但是只是實現(xiàn)了add , remove , fire , empty, has和帶參數(shù)的構(gòu)造函數(shù)功能,  $.Callbacks 還有disable,disabled, fireWith , fired , lock, locked 方法

 代碼如下:

 

復(fù)制代碼 代碼如下:

 String.prototype.trim = function ()
        {
            return this.replace( /^/s+|/s+$/g, '' );
        };

        // Simulate jQuery.Callbacks object
        function MyCallbacks( options )
        {
            var ops = { once: false, memory: false, unique: false, stopOnFalse: false };

            if ( typeof options === 'string' && options.trim() !== '' )
            {
                var opsArray = options.split( //s+/ );
                for ( var i = 0; i < options.length; i++ )
                {
                    if ( opsArray[i] === 'once' )
                        ops.once = true;
                    else if ( opsArray[i] === 'memory' )
                        ops.memory = true;
                    else if ( opsArray[i] === 'unique' )
                        ops.unique = true;
                    else if ( opsArray[i] === 'stopOnFalse' )
                        ops.stopOnFalse = true;
                }
            }

            var ar = [];
            var lastArgs = null;
            var firedTimes = 0;

            function hasName( name )
            {
                var h = false;

                if ( typeof name === 'string'
                    && name !== null
                    && name.trim() !== ''
                    && ar.length > 0 )
                {
                    for ( var i = 0; i < ar.length; i++ )
                    {
                        if ( ar[i].name === name )
                        {
                            h = true;
                            break;
                        }
                    }
                }

                return h;
            }

            // add a function
            this.add = function ( fn )
            {
                if ( typeof fn === 'function' )
                {
                    if ( ops.unique )
                    {
                        // check whether it had been added before
                        if ( fn.name !== '' && hasName( fn.name ) )
                        {
                            return this;
                        }
                    }

                    ar.push( fn );

                    if ( ops.memory )
                    {
                        // after added , call it immediately
                        fn.call( this, lastArgs );
                    }
                }

                return this;
            };

            // remove a function
            this.remove = function ( fn )
            {
                if ( typeof ( fn ) === 'function'
                    && fn.name !== ''
                    && ar.length > 0 )
                {
                    for ( var i = 0; i < ar.length; i++ )
                    {
                        if ( ar[i].name === fn.name )
                        {
                            ar.splice( i, 1 );
                        }
                    }
                }

                return this;
            };

            // remove all functions
            this.empty = function ()
            {
                ar.length = 0;
                return this;
            };

            // check whether it includes a specific function
            this.has = function ( fn )
            {
                var f = false;

                if ( typeof ( fn ) === 'function'
                    && fn.name !== ''
                    && ar.length > 0 )
                {
                    for ( var i = 0; i < ar.length; i++ )
                    {
                        if ( ar[i].name === fn.name )
                        {
                            f = true;
                            break;
                        }
                    }
                }

                return f;
            };

            // invoke funtions it includes one by one
            this.fire = function ( args )
            {
                if ( ops.once && firedTimes > 0 )
                {
                    return this;
                }

                if ( ar.length > 0 )
                {
                    var r;

                    for ( var i = 0; i < ar.length; i++ )
                    {
                        r = ar[i].call( this, args );

                        if ( ops.stopOnFalse && r === false )
                        {
                            break;
                        }
                    }
                }

                firedTimes++;

                if ( ops.memory )
                {
                    lastArgs = args;
                }

                return this;
            };
        };
 

 測試函數(shù)如下:(注意fn1 fn2是匿名函數(shù), fn2返回false , fn3是有“名”函數(shù))

 

復(fù)制代碼 代碼如下:

 var fn1 = function ( v )
        {
            console.log( 'fn1 ' + ( v || '' ) );
        };

        var fn2 = function ( v )
        {
            console.log( 'fn2 ' + ( v || '' ) );
            return false;
        };

        function fn3( v )
        {
            console.log( 'fn3 ' + ( v || '' ) );
        };
 

 1 . 測試add & fire

var cb=new MyCallbacks();

cb.add(fn1)

cb.add(fn2)

cb.add(fn3)

cb.fire('hello')

輸出:

fn1 hello
fn2 hello
fn3 hello

2.測試remove
var cb=new MyCallbacks();

cb.add(fn1)

cb.add(fn2)

cb.add(fn3)

cb.remove(fn1)
cb.fire('hello')
cb.remove(fn3)
cb.fire('hello')
輸出:

fn1 hello
fn2 hello
fn3 hello
----------------------------
fn1 hello
fn2 hello

2.測試has
var cb=new MyCallbacks();

cb.add(fn1)

cb.add(fn2)

cb.add(fn3)

cb.has(fn1) 

cb.has(fn3) 

輸出:

false

---------------

true

3.測試帶參數(shù)的構(gòu)造函數(shù) : once

var cb=new MyCallbacks('once')

cb.add(fn1)

cb.fire('hello')

cb.fire('hello')

cb.add(fn2)

cb.fire('hello')

輸出:

hello

-------------------

------------------

------------------------------

4.測試帶參數(shù)的構(gòu)造函數(shù) : memory

 var cb=new MyCallbacks('memory')

cb.add(fn1)

cb.fire('hello') // 輸出 : fn1 hello

cb.add(fn2) // 輸出 : fn2 hello

cb.fire('hello')

 輸出 :

 fn1 hello

 fn2 hello

5.測試帶參數(shù)的構(gòu)造函數(shù) : stopOnFalse

var cb=new MyCallbacks('stopOnFalse')

cb.add(fn1)

cb.add(fn2)

cb.add(fn3)

cb.fire('hello')

輸出:

fn1 hello
fn2 hello
6.測試帶參數(shù)的構(gòu)造函數(shù) :unique

var cb=new MyCallbacks('unique')

 

b.add(fn3)

b.add(fn3)

cb.fire('hello')

輸出:

fn3 hello

 

7. 測試帶組合參數(shù)的構(gòu)造函數(shù):四個設(shè)置參數(shù)可以隨意組合,一下只測試全部組合的情況, 不然要寫16個測試用例 T_T

var cb=new MyCallbacks('once memory unique stopOnFalse')

cb.add(fn1) // 輸出: fn1

cb.add(fn2) // 輸出: fn2

cb.add(fn3) //  輸出: fn3

cb.fire('hello')

輸出:

fn1 hello
fn2 hello
cb.fire('hello') // 輸出:沒有輸出

 

以下是官方API 文檔:

Description: A multi-purpose callbacks list object that provides a powerful way to manage callback lists.The $.Callbacks() function is internally used to provide the base functionality behind the jQuery $.ajax() and$.Deferred() components. It can be used as a similar base to define functionality for new components.

構(gòu)造函數(shù) : jQuery.Callbacks( flags )

flags
Type: String
An optional list of space-separated flags that change how the callback list behaves.
Possible flags:
once: Ensures the callback list can only be fired once (like a Deferred).
memory: Keeps track of previous values and will call any callback added after the list has been fired right away with the latest "memorized" values (like a Deferred).
unique: Ensures a callback can only be added once (so there are no duplicates in the list).
stopOnFalse: Interrupts callings when a callback returns false.
By default a callback list will act like an event callback list and can be "fired" multiple times.

Two specific methods were being used above: .add() and .fire(). The .add() method supports adding new callbacks to the callback list, while the .fire() method executes the added functions and provides a way to pass arguments to be processed by the callbacks in the same list.

利用Callbacks 實現(xiàn)發(fā)布訂閱模式 pub/sub: (官方文檔)

復(fù)制代碼 代碼如下:

var topics = {};

        jQuery.Topic = function ( id )
        {
            var callbacks,
                method,
                topic = id && topics[id];

            if ( !topic )
            {
                callbacks = jQuery.Callbacks();
                topic = {
                    publish: callbacks.fire,
                    subscribe: callbacks.add,
                    unsubscribe: callbacks.remove
                };
                if ( id )
                {
                    topics[id] = topic;
                }
            }
            return topic;
        };

使用

復(fù)制代碼 代碼如下:

$.Topic( 'mailArrived' ).subscribe( function ( e )
        {
            console.log( 'Your have new email! ' );
            console.log( "mail title : " + e.title );
            console.log( "mail content : " + e.content );
        }
        );

        $.Topic( 'mailArrived' ).publish( { title: 'mail title', content: 'mail content' } );


實現(xiàn)了其余的全部功能 :callbacks.disable , callbacks.disabled,   callbacks.fired,callbacks.fireWith, callbacks.lock, callbacks.locked ,然后重構(gòu)了下代碼結(jié)構(gòu), 將實現(xiàn)放入了匿名函數(shù)內(nèi), 然后通過工廠方法 window.callbacks 返回實例,以免每次使用必須 new .

具體代碼如下, 有興趣和時間的可以對照jQuery版本的Callbacks對比下 :

復(fù)制代碼 代碼如下:

( function ( window, undefined )
        {
            // Simulate jQuery.Callbacks object
            function Callbacks( options )
            {
                var ops = { once: false, memory: false, unique: false, stopOnFalse: false },
                    ar = [],
                    lastArgs = null,
                    firedTimes = 0,
                    _disabled = false,
                    _locked = false;

                if ( typeof options === 'string' && options.trim() !== '' )
                {
                    var opsArray = options.split( //s+/ );
                    for ( var i = 0; i < options.length; i++ )
                    {
                        if ( opsArray[i] === 'once' )
                            ops.once = true;
                        else if ( opsArray[i] === 'memory' )
                            ops.memory = true;
                        else if ( opsArray[i] === 'unique' )
                            ops.unique = true;
                        else if ( opsArray[i] === 'stopOnFalse' )
                            ops.stopOnFalse = true;
                    }
                }

                function hasName( name )
                {
                    var h = false;

                    if ( typeof name === 'string'
                        && name !== null
                        && name.trim() !== ''
                        && ar.length > 0 )
                    {
                        for ( var i = 0; i < ar.length; i++ )
                        {
                            if ( ar[i].name === name )
                            {
                                h = true;
                                break;
                            }
                        }
                    }

                    return h;
                }

                // add a function
                this.add = function ( fn )
                {
                    if ( typeof fn === 'function' )
                    {
                        if ( ops.unique )
                        {
                            // check whether it had been added before
                            if ( fn.name !== '' && hasName( fn.name ) )
                            {
                                return this;
                            }
                        }

                        ar.push( fn );

                        if ( ops.memory )
                        {
                            // after added , call it immediately
                            fn.call( this, lastArgs );
                        }
                    }

                    return this;
                };

                // remove a function
                this.remove = function ( fn )
                {
                    if ( typeof ( fn ) === 'function'
                        && fn.name !== ''
                        && ar.length > 0 )
                    {
                        for ( var i = 0; i < ar.length; i++ )
                        {
                            if ( ar[i].name === fn.name )
                            {
                                ar.splice( i, 1 );
                            }
                        }
                    }

                    return this;
                };

                // remove all functions
                this.empty = function ()
                {
                    ar.length = 0;
                    return this;
                };

                // check whether it includes a specific function
                this.has = function ( fn )
                {
                    var f = false;

                    if ( typeof ( fn ) === 'function'
                        && fn.name !== ''
                        && ar.length > 0 )
                    {
                        for ( var i = 0; i < ar.length; i++ )
                        {
                            if ( ar[i].name === fn.name )
                            {
                                f = true;
                                break;
                            }
                        }
                    }

                    return f;
                };

                this.disable = function ()
                {
                    _disabled = true;
                    return this;
                };

                this.disabled = function ()
                {
                    return _disabled;
                };

                this.fired = function ()
                {
                    return firedTimes > 0;
                };

                function _fire( context, args )
                {
                    if ( _disabled || ops.once && firedTimes > 0 || _locked )
                    {
                        return;
                    }

                    if ( ar.length > 0 )
                    {
                        var r;

                        for ( var i = 0; i < ar.length; i++ )
                        {
                            r = ar[i].call( context, args );

                            if ( ops.stopOnFalse && r === false )
                            {
                                break;
                            }
                        }
                    }

                    firedTimes++;

                    if ( ops.memory )
                    {
                        lastArgs = args;
                    }

                };

                this.fireWith = function ( context, args )
                {
                    context = context || this;
                    _fire( context, args );
                    return this;
                };

                this.fire = function ( args )
                {
                    _fire( this, args );
                    return this;
                };

                this.lock = function ()
                {
                    _locked = true;
                    return this;
                };

                this.locked = function ()
                {
                    return _locked;
                };

            };

            // exposed to global as a factory method
            window.callbacks = function ( options )
            {
                return new Callbacks( options );
            };

        } )( window );

發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
国产成人久久久久| 激情深爱一区二区| 天堂在线观看一卡二卡三卡四卡| 免费看毛片的网址| 亚洲欧美日韩国产手机在线| 一区二区三区激情视频| 天天噜噜噜噜噜噜| 成人免费淫片aa视频免费| 黄网视频在线观看| 久久99久久久久| 欧美成人性色生活仑片| 在线观看国产精品一区| juy有坂深雪中文字幕| 久久人妻少妇嫩草av无码专区| 欧美精品一区二区三区中文字幕| 91精品国产综合久久久久久豆腐| av不卡免费电影| 国产精品500部| 国产精品天美传媒| 四虎永久免费在线观看| 日韩一区二区三区视频在线观看| 高清av电影| 91国偷自产一区二区使用方法| 亚洲女同精品视频| 久久久久久蜜桃| 午夜精品在线观看| 福利一区和二区| 日韩网站中文字幕| 久久夜色精品一区| 国产精品自拍亚洲| 在线色视频观看| 亚洲欧美亚洲| 久久国产66| 羞羞视频在线观看免费| 91久久奴性调教| 日韩精品一二三四区| 成人一区视频| 成人中文字幕av| 精品免费一区二区三区| 亚洲999一在线观看www| 囯产精品久久久久久| 666av成人影院在线观看| 欧美专区一区二区三区| 中文字幕一区免费在线观看| 免费黄色三级网站| 午夜影院福利社| 久久久久免费看黄a片app| 免费网站www在线观看| 亚州精品国产| 蜜臀久久99精品久久久| 成人免费毛片aaaaa**| 久久久久性色av无码一区二区| 最新国产麻豆精品| 欧美激情第四页| 亚洲精品一区二区妖精| 欧美福利在线播放网址导航| 91麻豆精品一区二区三区| 亚洲成在线观看| 波多野结衣网页| 国产欧美在线观看| 欧美性xxxxx| 国模无码视频一区二区三区| 久久尤物视频| 一区二区三区精彩视频| 韩国av在线播放| 日韩av大片站长工具| 内射国产内射夫妻免费频道| av免费在线网址| 实拍女处破www免费看| 91麻豆国产自产在线观看| 亚洲天堂第二页| 国产黄色精品| 国产不卡一区| 久久先锋资源| 国产99久久精品一区二区 夜夜躁日日躁| 91丨国产丨九色丨pron| 欧美xxxxhdvideosex| 狠狠鲁男人天堂| 国产剧情演绎av| 国产精品激情自拍| 欧美a级免费视频| 自拍视频第一页| 久久国产精品久久久| 久久久久久国产视频| 国产精品一区二区久久久久| 国产精品久久久久久超碰| 五月天亚洲综合小说网| 色噜噜狠狠狠综合曰曰曰| 日韩国产大片| а√天堂www在线а√天堂视频| 国产美女久久久久| 国产精品加勒比| 精品欧美一区二区三区在线观看| 久久国产精品一区二区三区| h视频网站在线观看| 欧美日韩综合高清一区二区| 国产淫片在线观看| 精品国产无码一区二区三区| 在线观看亚洲网站| 国产午夜精品久久久| 深夜福利免费在线观看| 99热在线免费| 国产美女被草| 欧美区一区二区| 国产精品区在线观看| 蜜臀91精品一区二区三区| 欧美在线视频网| 特级西西444www大精品视频免费看| 精精国产xxx在线视频app| a级国产乱理论片在线观看99| 国产精品ⅴa在线观看h| 亚洲三级色网| 欧美黑人巨大xxxxx| 欧美亚日韩国产aⅴ精品中极品| 精品国产依人香蕉在线精品| 亚洲超碰在线| 国产亚洲免费的视频看| 大肉大捧一进一出好爽动态图| 99免费视频观看| 国产视频一区在线播放| 欧美肉大捧一进一出免费视频| 日韩免费视频播放| 亚洲精品国产嫩草在线观看| 97青娱国产盛宴精品视频| 精品久久久久久一区| 91九色蝌蚪| 欧美做受高潮6| 欧美大香线蕉线伊人久久| 国产黑人绿帽在线第一区| 久久女人天堂| 草莓视频末满18勿| 扒开腿狂躁女人爽出白浆2| 亚洲永久无码7777kkk| 波多野结衣一区二区在线| 欧美日韩国产精品一区二区亚洲| 久久黄色美女电影| 1024精品久久久久久久久| 欧美高清视频在线高清观看mv色露露十八| 日本aⅴ亚洲精品中文乱码| √新版天堂资源在线资源| 免费高清视频精品| 蜜桃视频一区二区在线观看| 欧美 日韩 国产精品| 91久久精品日日躁夜夜躁欧美| 亚洲国产果冻传媒av在线观看| 国产精品一站二站| av毛片在线免费观看| 亚洲成a人片| 五月婷婷开心网| 欧美精品momsxxx| 午夜精彩视频在线观看不卡| 欧美一区免费视频| 不卡一区二区三区视频| h视频在线观看免费完整版| 国产麻豆一区二区三区| 97超超碰碰| 一区二区三区在线播放视频| 日本精品视频在线播放| 福利精品视频| 国产日产精品一区| 992tv在线观看| 欧美日韩在线观看一区二区| 午夜久久资源| 又黄又爽毛片免费观看| 午夜精品一区二区三| 色偷偷亚洲第一成人综合网址| 国产一级性生活| 欧美成人一二区| xxxxhd欧美精品| 日本久久久精品视频| 男人艹女人在线观看| 不卡的av电影在线观看| 日韩精品久久久久久久的张开腿让| eeuss影院18www免费| 国产精品久久久久久久天堂第1集| 国产91精品久久久久久久| 丝袜国产免费观看| 久久精品一偷一偷国产| 日韩中文在线中文网在线观看| 国产一区视频免费观看| 精品一区二区三区久久久| 国产h视频在线观看| 无码人妻久久一区二区三区| 国产精品视频一区二区三区四区五区| 久久久久久久久影院| 欧美黄色成人网| 天堂精品高清1区2区3区| 欧美在线日韩精品| 日本中文字幕免费在线观看| 国产午夜福利一区二区| 久久蜜桃av| 日韩激情综合网| 日本久久精品一区二区| 国产一本一道久久香蕉| 欧美夫妻性视频| 欧美最大成人综合网| 午夜精品一区二区三区视频免费看| 8x8x成人免费视频| 精品久久一区二区| 精品国产精品一区二区夜夜嗨| 比比资源-先锋影音资源站| 五福影院新址进入www1378| 国产一区二区三区精品久久久| 日韩精品在线免费视频| 蜜桃91在线| 四虎永久国产精品| 国产精品久久久久久久久久10秀| 国产福利一区二区三区在线观看| 国产亚洲字幕| 97成人在线观看视频| 青青视频一区二区| 久久午夜羞羞影院免费观看| 久久国产三级精品| 国产成人精品免费视频网站| 亚洲欧洲在线一区| 一二三四在线观看免费高清中文在线观看| 国产女人高潮的av毛片| 亚洲黄网在线观看| 99久久综合99久久综合网站| 日韩激情视频在线观看| 亚洲一区一卡| 精品久久综合| 欧美视频一区二区| 国产福利a级| 中国女人内谢25xxxxx| 一区二区激情视频| 色综合av综合无码综合网站| 蜜臀尤物一区二区三区直播| 天天草天天爽| 亚洲精品电影在线观看| 中文字幕日韩在线视频| fc2在线中文字幕| 国产一区二区精品久| 男女激情无遮挡| 黄色高清视频| 白白色视频在线| 亚洲国产精品无码久久| 成年免费网站| 亚洲一区二区三区免费观看| 国产最新视频在线| 欧美午夜精品久久久久免费视| 色噜噜一区二区| 欧美高清另类hdvideosexjaⅴ| 韩国三级电影在线观看婷婷| 九九九视频在线观看| 五月激激激综合网色播| 明星裸体视频一区二区| 黄网站免费在线播放| 狠狠一区二区三区| 国产精品视频看看| 欧美挤奶吃奶水xxxxx| rebdb初裸写真在线观看| 欧美日韩在线免费观看视频| 国产精品一区二区三区在线免费观看| 久久久精品在线观看| 最新中文字幕视频| 亚洲精品www久久久久久广东| 美女亚洲一区| 欧美精品v日韩精品v韩国精品v| 天天综合日日夜夜精品| 国产美女福利在线观看| 一区二区免费在线观看视频| 欧美黄页在线免费观看| 欧美大胆成人| 成人福利小视频| 国产 欧美 日韩 在线| 婷婷综合久久中文字幕蜜桃三电影| 99热这里只有精品66| 精品在线视频观看| 美女呻吟一区| 午夜精品一区二区三区四区| 香蕉久久国产av一区二区| 中文字幕乱码在线| 99视频在线免费播放| 欧美bbb人妖| 男女羞羞免费视频| 久久精品日韩精品| 国产伦理一区二区三区| 欧美久草视频| 久久免费99精品久久久久久| 亚洲综合男人的天堂| 亚洲精品一区二区妖精| 久久在线中文字幕| 最新中文乱码字字幕在线| 亚洲永久免费精品| 国产一区二区日韩精品欧美精品| 欧美成人午夜视频| 国产精品毛片a∨一区二区三区| 成人精品动漫一区二区三区| 国产欧美精品在线| 亚洲最大福利视频网| 久cao在线| 国产丝袜视频在线观看| 亚洲欧美综合久久久| 国产第一页在线视频| 男女av在线| 最新黄色av网站| 国产精品久久久久9999| 久草中文综合在线| 国内爆初菊对白视频| 蜜臀久久99精品久久久无需会员| 欧美大尺度做爰床戏| 国产一区二区精品久| 国产不卡一卡2卡三卡4卡5卡在线| 奇米777国产一区国产二区| 久久狠狠亚洲综合| 亚洲av无码乱码国产麻豆| 韩日毛片在线观看| 奇米影视一区二区三区| 尤物视频一区二区| 欧美洲成人男女午夜视频| 三级毛片在线看| 成人在线直播| 欧美美女激情18p| 日韩欧美综合| 日韩一区二区欧美| 久久精品无码专区| 国产美女一区二区三区| 性xxxxxxxxx| 精品国产乱码久久久久久郑州公司| 欧美激情久久久久久久| 欧美专区一二三| 久久夜色精品亚洲噜噜国产mv| 丰满人妻一区二区三区无码av| 白丝女仆被免费网站| 99riav视频在线观看| 日韩av电影在线观看| 成人精品国产福利|