JavaScript如何一次性展示幾萬條數(shù)據(jù)

通過DOM操作插入到頁面熟丸,勢必導致頁面運行出現(xiàn)卡頓, 為此我還特意寫了一個 demo測試了一下伪节, 代碼如下

//data.json
var data=[{"title":"標題","content":"hahahahaha"},]
 <script src="https://cdn.staticfile.org/jquery/1.10.2/jquery.min.js"> </script>
<script src="./data.json"></script>
<script>
 loadAll( data);
function loadAll(response) {
    var html = "";
    for (var i = 0; i < response.length; i++) {
        var item = response[i];
        html += "<li>title:" + item.title + " content:" + item.content + "</li>";
    }
    $("#content").html(html);
}
</script>

data.json中大概有13萬條數(shù)據(jù)左右光羞, 通過ajax獲取數(shù)據(jù)后以最簡單粗暴的方法展示數(shù)據(jù),在chrome瀏覽器下怀大, 刷新頁面到數(shù)據(jù)顯示雏逾,我心中默數(shù)防泵, 整個過程大概花掉5秒鐘左右的時間, 卡頓非常明顯。 我大致觀察了一下代碼的運行時間馏鹤,發(fā)現(xiàn)循環(huán)生成字符串這過程其實并不算太耗時, 性能瓶頸是在將html字符串插入到文檔中這個過程上蓉坎, 也就是 $("#content").html(html); 這句代碼的執(zhí)行黎烈, 畢竟有13萬個li元素要被挺入到文檔里面, 頁面渲染速度緩慢也在情理之中蒜焊。

既然一次渲染13萬條數(shù)據(jù)會造成頁面加載速度緩慢倒信,那么我們可以不要一次性渲染這么多數(shù)據(jù),而是分批次渲染泳梆, 比如一次10000條鳖悠,分13次來完成, 這樣或許會對頁面的渲染速度有提升优妙。 然而乘综,如果這13次操作在同一個代碼執(zhí)行流程中運行,那似乎不但無法解決糟糕的頁面卡頓問題套硼,反而會將代碼復雜化卡辰。 類似的問題在其它語言最佳的解決方案是使用多線程,JavaScript雖然沒有多線程熟菲,但是setTimeout和setInterval兩個函數(shù)卻能起到和多線程差不多的效果看政。 因此,要解決這個問題抄罕, 其中的setTimeout便可以大顯身手允蚣。 setTimeout函數(shù)的功能可以看作是在指定時間之后啟動一個新的線程來完成任務。

 loadAll( data);
function loadAll(response) {
    //將13萬條數(shù)據(jù)分組呆贿, 每組500條嚷兔,一共260組
    var groups = group(response);
    for (var i = 0; i < groups.length; i++) {
        //閉包森渐, 保持i值的正確性
        window.setTimeout(function () {
            var group = groups[i];
            var index = i + 1;
            return function () {
                //分批渲染
                loadPart( group, index );
            }
        }(), 1);
    }
}
 
//數(shù)據(jù)分組函數(shù)(每組500條)
function group(data) {
    var result = [];
    var groupItem;
    for (var i = 0; i < data.length; i++) {
        if (i % 500 == 0) {
            groupItem != null && result.push(groupItem);
            groupItem = [];
        }
        groupItem.push(data[i]);
    }
    result.push(groupItem);
    return result;
}
 
var currIndex = 0;
 
//加載某一批數(shù)據(jù)的函數(shù)
function loadPart( group, index ) {
    var html = "";
    for (var i = 0; i < group.length; i++) {
        var item = group[i];
        html += "<li>title:" + item.title + index + " content:" + item.content + index + "</li>";
    }
    //保證順序不錯亂
    while (index - currIndex == 1) {
        $("#content").append(html);
        currIndex = index;
    }
}

以上代碼大致的執(zhí)行流程是

  1. 用ajax獲取到需要處理的數(shù)據(jù), 共13萬條
  2. 將數(shù)組分組冒晰,每組500條同衣,一共260組
  3. 循環(huán)這260組數(shù)據(jù),分別處理每一組數(shù)據(jù)壶运, 利用setTimeout函數(shù)開啟一個新的執(zhí)行線程(異步)耐齐,防止主線程因渲染大量數(shù)據(jù)導致阻塞。

loadPart函數(shù)中有這段代碼

while (index - currIndex == 1) {
    $("#content").append(html);
    currIndex = index;
}

是為了保證不同的線程中最終插入html到文檔中時順序的一致性蒋情, 不至于同時執(zhí)行的代碼在插入html時互相篡位埠况。

通過這種方式執(zhí)行, 頁面瞬間就刷出來了棵癣,不用絲毫等待時間辕翰。 從同步改為異步,雖然代碼的整體資源消耗增加了狈谊, 但是頁面卻能瞬間響應喜命, 而且, 前端的運行環(huán)境是用戶的電腦河劝,因此些許的性能損失帶來的用戶體驗提升相對來說還是值得的壁榕。

雖然示例中提到的情況在現(xiàn)實環(huán)境中幾乎不可能出現(xiàn), 但是在我們平時的工作中總會有一些似是而非的場景出現(xiàn)赎瞎, 利用里面的處理思路护桦, 或許對我們解決問題會有一定的幫助。

ps:setTimeout并不算真正的多線程煎娇, 但是為了方便表達,便借用了線程一詞

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末贪染,一起剝皮案震驚了整個濱河市缓呛,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌杭隙,老刑警劉巖哟绊,帶你破解...
    沈念sama閱讀 212,718評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異痰憎,居然都是意外死亡票髓,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,683評論 3 385
  • 文/潘曉璐 我一進店門铣耘,熙熙樓的掌柜王于貴愁眉苦臉地迎上來洽沟,“玉大人,你說我怎么就攤上這事蜗细●刹伲” “怎么了怒详?”我有些...
    開封第一講書人閱讀 158,207評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長踪区。 經(jīng)常有香客問我昆烁,道長,這世上最難降的妖魔是什么缎岗? 我笑而不...
    開封第一講書人閱讀 56,755評論 1 284
  • 正文 為了忘掉前任静尼,我火速辦了婚禮,結果婚禮上传泊,老公的妹妹穿的比我還像新娘鼠渺。我一直安慰自己,他們只是感情好或渤,可當我...
    茶點故事閱讀 65,862評論 6 386
  • 文/花漫 我一把揭開白布系冗。 她就那樣靜靜地躺著,像睡著了一般薪鹦。 火紅的嫁衣襯著肌膚如雪掌敬。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 50,050評論 1 291
  • 那天池磁,我揣著相機與錄音奔害,去河邊找鬼。 笑死地熄,一個胖子當著我的面吹牛华临,可吹牛的內容都是我干的。 我是一名探鬼主播端考,決...
    沈念sama閱讀 39,136評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼雅潭,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了却特?” 一聲冷哼從身側響起扶供,我...
    開封第一講書人閱讀 37,882評論 0 268
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎裂明,沒想到半個月后椿浓,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,330評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡闽晦,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,651評論 2 327
  • 正文 我和宋清朗相戀三年扳碍,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片仙蛉。...
    茶點故事閱讀 38,789評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡笋敞,死狀恐怖,靈堂內的尸體忽然破棺而出荠瘪,到底是詐尸還是另有隱情液样,我是刑警寧澤振亮,帶...
    沈念sama閱讀 34,477評論 4 333
  • 正文 年R本政府宣布,位于F島的核電站鞭莽,受9級特大地震影響坊秸,放射性物質發(fā)生泄漏。R本人自食惡果不足惜澎怒,卻給世界環(huán)境...
    茶點故事閱讀 40,135評論 3 317
  • 文/蒙蒙 一褒搔、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧喷面,春花似錦星瘾、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,864評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至盒齿,卻和暖如春念逞,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背边翁。 一陣腳步聲響...
    開封第一講書人閱讀 32,099評論 1 267
  • 我被黑心中介騙來泰國打工翎承, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人符匾。 一個月前我還...
    沈念sama閱讀 46,598評論 2 362
  • 正文 我出身青樓叨咖,卻偏偏與公主長得像,于是被迫代替她去往敵國和親啊胶。 傳聞我的和親對象是個殘疾皇子甸各,可洞房花燭夜當晚...
    茶點故事閱讀 43,697評論 2 351

推薦閱讀更多精彩內容