自定義事件解決重覆請求BUG

来源:http://www.cnblogs.com/LuckyWinty/archive/2017/07/10/7137538.html
-Advertisement-
Play Games

現在,組件化開發還是比較流行的,畢竟其優點相當突出。最近在開發一個組件的時候,遇到了一個很有意思的BUG。。。 BUG的背景 最近在開發一個組件,好不容易開發好了轉測試。然後,測試給我提了一個這樣的bug,orz... 因為是一個組件,最大的好處就是可以隨處復用,隨處使用,然而,當一個頁面用了多個組 ...


現在,組件化開發還是比較流行的,畢竟其優點相當突出。最近在開發一個組件的時候,遇到了一個很有意思的BUG。。。

 

BUG的背景

  最近在開發一個組件,好不容易開發好了轉測試。然後,測試給我提了一個這樣的bug,orz...

因為是一個組件,最大的好處就是可以隨處復用,隨處使用,然而,當一個頁面用了多個組件,只有最後一個生效的時候,這個組件就沒有什麼意義了。。。

 

BUG原因查找

  這個組件的初始數據來源的介面是固定的,也就是說,頁面內的所有這個組件在初始化的時候都會發出同樣的請求,這裡的請求是jsonp的方式,所以回調函數是綁定在window上的一個函數,但是在頁面中window只有一個,所以在回調處理的時候,要處理的組件內的相應的數據只指向最後一個組件。所以導致多個同樣的組件在同一個頁面中,只有最後一個組件能在取得數據之後順利渲染出來。

 

BUG解決思路

  最主要就是要將每次請求的callback存儲起來,這樣就可以保證callback中對組件數據的處理不是只指向最後一個。其次,既然是一樣的請求,當然不希望會發出兩次以上啦,即一個頁面發出的每一個請求都是唯一的。

 

BUG解決方案

  想到了發佈訂閱者模式的自定義事件,可以寫這樣的一個模塊,每次請求發出前判斷一下之前是否有相同的模塊已經發出了,如果沒有則緩存callback發出請求,如果有相同的請求已經發出了,那麼檢查一下這個發出的請求是否已經完成了,如果沒有則繼續緩存callback等待,如果請求已經發出並且已經完成則直接處理callback。在請求第一次回來後,發出廣播,把之前緩存的callback都執行一次。

 

自定義事件詳情

  定義一個模塊,裡面有n個以回調函數命名的事件對象,每個對象有在被初始化的時候,定義其狀態state,對應的callback數組,請求回到的數據data。每次調用該模塊,首先檢查對應的cbName是否被初始化,然後檢查其state。根據state做相應的操作並改變state的值。state的值有3中,分別為init、loading、loaded。即初始化、請求中、請求完成。處於請求完成狀態時才能執行相應的回調。具體如下:

define('wq.getData', function (require, exports, module) {
    var ls = require('loadJs');
    
    var cache = {};
    cache.init = function(cb,cbName,url){
        if(!cache[cbName]){
            cache[cbName] = {};
            cache[cbName].state = 'init';
            cache[cbName].cbs = [];
            cache[cbName].data = [];
        }
        cache.on(cb,cbName,url);
    }
    cache.on = function(cb,cbName,url){
        if(cache[cbName].state == 'loaded'){
            cb(cache[cbName].data)
        }else if(cache[cbName].state == 'loading'){
            cache[cbName].cbs.push(cb)
        }else if(cache[cbName].state == 'init'){
            cache[cbName].cbs.push(cb);
            cache[cbName].state = 'loading';
            cache.fetch(cb,cbName,url);
        }
    }
    cache.broadcast = function(cbName){
        cache[cbName].cbs.forEach(function(cb){
            cb(cache[cbName].data)
        });
    }
    cache.checkLoaded = function(cbName){
        if(cache[cbName].data[0]){
            cache[cbName].state = 'loaded';
            cache.broadcast(cbName);
        }
    }
    cache.fetch = function(cb,cbName,url){
        ls.loadScript({
            url: url,
            charset: 'utf-8',
            handleError:function(func, args, context,errorObj){
                console.log(_errlogText + context);
                cache[cbName].data[0] = {};
                cache.checkLoaded(cbName);
            }
        });
        if(window.cbName) return;
        window[cbName] = function(json){
            cache[cbName].data[0] = json;
            cache.checkLoaded(cbName);
        }
    }

    exports.getData = function(cb,cbName,url){
        cache.init(cb,cbName,url);
    }  

})

完美解決問題,每個回調都不會遺漏或者被覆蓋……

 

擴展思路

       該模塊可通用於處理一個頁面內同一個請求的情況。還可以擴展到處理一些需要2個請求以上完成才執行某個回調的情況。類似於Promose的情況。這個時候可以規定,每個data[0]裝的是固定的對應介面的數據,data[2]對應另一個,一次類推。不過這樣就要遍歷到每一項都為true的時候才執行回調。而且對應關係比較容易混亂,再擴展就不如直接用Promise來處理了。。。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 手頭現在有一份福布斯2016年全球上市企業2000強排行榜的數據,但原始數據並不規範,需要處理後才能進一步使用。 本文通過實例操作來介紹用pandas進行數據整理。 ...
  • 首先,萬分抱歉,據上一篇更新後,時隔已近一個月。雖然博主不日念起此事,但實在是最近繁事纏身,說這些也是想告訴對此文有所期待的朋友,博主一定會更完到最後一章,不盡人意之處,還請各位多擔待。如果不出意外,博主還是努力一周更兩篇。 前面,我們看過項目實際運行的功能介紹,表結構也有所瞭解。今天,我們就正式進 ...
  • 時間限制: 1 s 空間限制: 128000 KB 題目等級 : 鑽石 Diamond 題解 查看運行結果 時間限制: 1 s 空間限制: 128000 KB 題目等級 : 鑽石 Diamond 時間限制: 1 s 空間限制: 128000 KB 題目等級 : 鑽石 Diamond 時間限制: 1 ...
  • 為啥整合 Dubbo 實現 SOA Dubbo 不單單隻是高性能的 RPC 調用框架,更是 SOA 服務治理的一種方案。 ...
  • 之前寫過在windows環境上部署rabbitmq,這回介紹在centos上對這個消息中間件進行部署的過程 一 下載和解壓 wget http://www.rabbitmq.com/releases/rabbitmq-server/current/rabbitmq-server_3.6.10-1.d ...
  • // test07.cpp : Defines the entry point for the console application.// #include "stdafx.h"//設計模式第7章 適配器模式 class Duck{public: virtual void quack() = 0; ...
  • 創建資料庫(y2165) MyBatis環境搭建1.在pom.xml引入依賴2.得替換build節點,為了讓程式編譯在main中所有子包下的配置文件3.構建大配置,位於resources<?xml version="1.0" encoding="UTF-8"?><!DOCTYPE configura ...
  • ngrx 是 Angular框架的狀態容器,提供可預測化的狀態管理。 1.首先創建一個可路由訪問的模塊 這裡命名為:DemopetModule。 包括文件:demopet.html、demopet.scss、demopet.component.ts、demopet.routes.ts、demopet ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...