Heim > Web-Frontend > js-Tutorial > Geben Sie einen gekapselten Javascript-Ereigniswarteschlangen-Funktionscode frei, um das Problem der Bindung von Ereignissen zu lösen

Geben Sie einen gekapselten Javascript-Ereigniswarteschlangen-Funktionscode frei, um das Problem der Bindung von Ereignissen zu lösen

伊谢尔伦
Freigeben: 2017-07-22 11:01:03
Original
1772 Leute haben es durchsucht

Bei der Verwendung von addEventListener() oder attachmentEvent() in JavaScript zum Binden von Ereignissen treten mehrere kleinere Probleme auf:

1. Anonyme Funktionen, die mit addEventListener() oder attachmentEvent() hinzugefügt werden, können nicht sein ENTFERNT.

var oBtn = document.getElementById('btn');
oBtn.addEventListener('click',function(){
    alert('button is clicked')
},false)
oBtn.reomveEventListener('click',function(){
    alert('button is clicked')
},false)
//oBtn上的事件无法移除,因为传入的是一个匿名函数
Nach dem Login kopieren

2. In ie6-ie8 das Ausführungsproblem in umgekehrter Reihenfolge bei der Verwendung von attachmentEvent() zum Binden mehrerer Ereignisse.

var oBtn = document.getElementById('btn');
oBtn.attachEvent('onclick',function(){
    alert(1)
})
oBtn.attachEvent('onclick',function(){
    alert(2)
})
oBtn.attachEvent('onclick',function(){
    alert(3)
})
//ie9+   下执行顺序1、2、3
//ie6-ie8下执行顺序3、2、1
Nach dem Login kopieren

Das Problem lösen

Ich möchte ein browserübergreifendes Ereignisbindungsmodul schreiben, damit es in Zukunft wiederverwendet werden kann. Ich möchte das Berufungsproblem lösen. JQuery verwendet intern Ereigniswarteschlangen und Daten-Caching-Mechanismen, um dieses Problem zu lösen. Ich habe mir den entsprechenden Quellcode angesehen und festgestellt, dass es wirklich kompliziert ist. Ich habe selbst einige Methoden ausprobiert und es geschafft. Der von mir gepostete Codeausschnitt war ursprünglich objektorientiert geschrieben. Ich wollte ihn nicht kompliziert machen, deshalb habe ich ihn in eine Funktion geändert, um ihn zu organisieren.

/*绑定事件的接口
 *
 *@param    {dom-DOM}和{type-string}和{fn-function}  可选参数{fnName-string}
 *@execute  创建事件队列,添加到DOM对象属性上,
            将事件处理程序(函数)加入事件队列
            可为事件处理程序添加一个标识符,用于删除指定事件处理程序
  */
 function bind(dom,type,fn,fnName){
    dom.eventQueue = dom.eventQueue || {};
    dom.eventQueue[type] = dom.eventQueue[type] || {};
    dom.handler = dom.handler || {};
    if (!fnName) {
        var index = queueLength(dom,type);
        dom.eventQueue[type]['fnQueue'+index] = fn;
    }
    else {
        dom.eventQueue[type][fnName] = fn;
    };
    if (!dom.handler[type]) bindEvent(dom,type);
};
/*绑定事件
 *
 *@param    {dom-DOM}和{type-string}
 *@execute  只绑定一次事件,handler用于遍历执行事件队列中的事件处理程序(函数)
 *@caller   bind()
 */
function bindEvent(dom,type){
    dom.handler[type] = function(){
        for(var guid in dom.eventQueue[type]){
            dom.eventQueue[type][guid].call(dom);
        }
    };
    if (window.addEventListener) {
        dom.addEventListener(type,dom.handler[type],false);
    }
    else {
        dom.attachEvent('on'+type,dom.handler[type]);
    };
};
/*移除事件的接口
 *
 *@param    {dom-DOM}和{type-string} 可选参数{fnName-function}
 *@execute  如果没有标识符,则执行unBindEvent()
            如果有标识符,则删除指定事件处理程序,如果事件队列长度为0,执行unBindEvent()
  */
function unBind(dom,type,fnName){
    var hasQueue = dom.eventQueue && dom.eventQueue[type];
    if (!hasQueue) return;
    if (!fnName) {
        unBindEvent(dom,type)
    }
    else {
        delete dom.eventQueue[type][fnName];
        if (queueLength(dom,type) == 0) unBindEvent(dom,type);
    };
};
/*移除事件
 *
 *@param    {dom-DOM}和{type-string}
 *@execute  移除绑定的事件处理程序handler,并清空事件队列
 *@caller   unBind()
 */
function unBindEvent(dom,type){
    if (window.removeEventListener) {
        dom.removeEventListener(type,dom.handler[type])
    }
    else {
        dom.detachEvent(type,dom.handler[type])
    }
    delete dom.eventQueue[type];
};
/*判断事件队列长度
 *
 *@param    {dom-DOM}和{type-string}
 *@caller   bind() unBind()
 */
function queueLength(dom,type){
    var index = 0;
    for (var length in dom.eventQueue[type]){
        index++ ;
    }
    return index;
};
Nach dem Login kopieren

Anwendung

var oBtn = document.getElementById('btn');
//绑定事件
//为button同时绑定三个click事件函数
//ie6-ie8下执行顺序不变
bind(oBtn,'click',function(){
    alert(1);
})
bind(oBtn,'click',function(){
    alert(2);
},'myFn')
bind(oBtn,'click',function(){
    alert(3);
})
//移除事件
//移除所有绑定的click事件函数,支持移除匿名函数
unBind(oBtn,'click')
//只移除标识符为myfn的事件函数
unBind(oBtn,'click','myFn')
Nach dem Login kopieren

Das obige ist der detaillierte Inhalt vonGeben Sie einen gekapselten Javascript-Ereigniswarteschlangen-Funktionscode frei, um das Problem der Bindung von Ereignissen zu lösen. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Verwandte Etiketten:
Quelle:php.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage