摘要:即在回調(diào)被執(zhí)行前,多次調(diào)用帶有同一回調(diào)函數(shù)的,會導致回調(diào)在同一幀中執(zhí)行多次。例子中的是由傳給回調(diào)函數(shù)的,表示回調(diào)隊列被觸發(fā)的時間。完美的解決方案是通過來管理隊列,其思路就是保證的隊列里,同樣的回調(diào)函數(shù)只有一個。
requestAnimationFrame 方法讓我們可以在下一幀開始時調(diào)用指定函數(shù)。但是很多人可能不知道,不管三七二十一直接在 requestAnimationFrame 的回調(diào)函數(shù)里繪制動畫會有一個問題。是什么問題呢?要理解這個問題,我們先要了解 requestAnimationFrame 的一個知識點。
requestAnimationFrame 不管理回調(diào)函數(shù)這個知識點就是 requestAnimationFrame 不管理回調(diào)函數(shù)。這一點在 w3c 中明確說明了。
Also note that multiple calls to requestAnimationFrame with the same callback (before callbacks are invoked and the list is cleared) will result in multiple entries being in the list with that same callback, and thus will result in that callback being invoked more than once for the animation frame.
— w3c
即在回調(diào)被執(zhí)行前,多次調(diào)用帶有同一回調(diào)函數(shù)的 requestAnimationFrame,會導致回調(diào)在同一幀中執(zhí)行多次。我們可以通過一個簡單的例子模擬在同一幀內(nèi)多次調(diào)用 requestAnimationFrame 的場景:
const animation = timestamp => console.log("animation called at", timestamp) window.requestAnimationFrame(animation) window.requestAnimationFrame(animation) // animation called at 320.7559999991645 // animation called at 320.7559999991645
我們用連續(xù)調(diào)用兩次 requestAnimationFrame 模擬在同一幀中調(diào)用兩次 requestAnimationFrame。
例子中的 timestamp 是由 requestAnimationFrame 傳給回調(diào)函數(shù)的,表示回調(diào)隊列被觸發(fā)的時間。由輸出可知,animation 函數(shù)在同一幀內(nèi)被執(zhí)行了兩次,即繪制了兩次動畫。然而在同一幀繪制兩次動畫很明顯是多余的,相當于畫了一幅畫,然后再在這幅畫上再畫上同樣的一幅畫。
問題那么什么場景下,requestAnimationFrame 會在一幀內(nèi)被多次調(diào)用呢?熟悉事件的同學應該馬上能想到 mousemove, scroll 這類事件。
所以前面我們提到的問題就是:因為 requestAnimationFrame 不管理回調(diào)函數(shù),在滾動、觸摸這類高觸發(fā)頻率的事件回調(diào)里,如果調(diào)用 requestAnimationFrame 然后繪制動畫,可能會造成多余的計算和繪制。例如:
window.addEventListener("scroll", e => { window.requestAnimationFrame(timestamp => { animation(timestamp) }) })
在上面代碼中,scroll 事件可能在一幀內(nèi)多次觸發(fā),所以 animation 函數(shù)可能會在一幀內(nèi)重復繪制,造成不必要的計算和渲染。
解決方法對于這種高頻發(fā)事件,一般的解決方法是使用節(jié)流函數(shù)。但是在這里使用節(jié)流函數(shù)并不能完美解決問題。因為節(jié)流函數(shù)是通過時間管理隊列的,而 requestAnimationFrame 的觸發(fā)時間是不固定的,在高刷新頻率的顯示屏上時間會小于 16.67ms,頁面如果被推入后臺,時間可能大于 16.67ms。
完美的解決方案是通過 requestAnimationFrame 來管理隊列,其思路就是保證 requestAnimationFrame 的隊列里,同樣的回調(diào)函數(shù)只有一個。示意代碼如下:
const onScroll = e => { if (scheduledAnimationFrame) { return } scheduledAnimationFrame = true window.requestAnimationFrame(timestamp => { scheduledAnimationFrame = false animation(timestamp) }) } window.addEventListener("scroll", onScroll)
但是每次都要寫這么一堆代碼,也有點麻煩。所以我開源了 raf-plus 庫用于解決這個問題,有需要的的同學可以用用~
結(jié)論requestAnimationFrame 不管理回調(diào)函數(shù)隊列,而滾動、觸摸這類高觸發(fā)頻率事件的回調(diào)可能會在同一幀內(nèi)觸發(fā)多次。所以正確使用 requestAnimationFrame 的姿勢是,在同一幀內(nèi)可能調(diào)用多次 requestAnimationFrame 時,要管理回調(diào)函數(shù),防止重復繪制動畫。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/51004.html
摘要:即在回調(diào)被執(zhí)行前,多次調(diào)用帶有同一回調(diào)函數(shù)的,會導致回調(diào)在同一幀中執(zhí)行多次。例子中的是由傳給回調(diào)函數(shù)的,表示回調(diào)隊列被觸發(fā)的時間。完美的解決方案是通過來管理隊列,其思路就是保證的隊列里,同樣的回調(diào)函數(shù)只有一個。 requestAnimationFrame 方法讓我們可以在下一幀開始時調(diào)用指定函數(shù)。但是很多人可能不知道,不管三七二十一直接在 requestAnimationFrame 的...
摘要:即在回調(diào)被執(zhí)行前,多次調(diào)用帶有同一回調(diào)函數(shù)的,會導致回調(diào)在同一幀中執(zhí)行多次。例子中的是由傳給回調(diào)函數(shù)的,表示回調(diào)隊列被觸發(fā)的時間。完美的解決方案是通過來管理隊列,其思路就是保證的隊列里,同樣的回調(diào)函數(shù)只有一個。 requestAnimationFrame 方法讓我們可以在下一幀開始時調(diào)用指定函數(shù)。但是很多人可能不知道,不管三七二十一直接在 requestAnimationFrame 的...
摘要:即在回調(diào)被執(zhí)行前,多次調(diào)用帶有同一回調(diào)函數(shù)的,會導致回調(diào)在同一幀中執(zhí)行多次。例子中的是由傳給回調(diào)函數(shù)的,表示回調(diào)隊列被觸發(fā)的時間。完美的解決方案是通過來管理隊列,其思路就是保證的隊列里,同樣的回調(diào)函數(shù)只有一個。 requestAnimationFrame 方法讓我們可以在下一幀開始時調(diào)用指定函數(shù)。但是很多人可能不知道,不管三七二十一直接在 requestAnimationFrame 的...
摘要:即在回調(diào)被執(zhí)行前,多次調(diào)用帶有同一回調(diào)函數(shù)的,會導致回調(diào)在同一幀中執(zhí)行多次。例子中的是由傳給回調(diào)函數(shù)的,表示回調(diào)隊列被觸發(fā)的時間。完美的解決方案是通過來管理隊列,其思路就是保證的隊列里,同樣的回調(diào)函數(shù)只有一個。 requestAnimationFrame 方法讓我們可以在下一幀開始時調(diào)用指定函數(shù)。但是很多人可能不知道,不管三七二十一直接在 requestAnimationFrame 的...
摘要:即在回調(diào)被執(zhí)行前,多次調(diào)用帶有同一回調(diào)函數(shù)的,會導致回調(diào)在同一幀中執(zhí)行多次。例子中的是由傳給回調(diào)函數(shù)的,表示回調(diào)隊列被觸發(fā)的時間。完美的解決方案是通過來管理隊列,其思路就是保證的隊列里,同樣的回調(diào)函數(shù)只有一個。 requestAnimationFrame 方法讓我們可以在下一幀開始時調(diào)用指定函數(shù)。但是很多人可能不知道,不管三七二十一直接在 requestAnimationFrame 的...
閱讀 2422·2021-11-25 09:43
閱讀 1202·2021-09-07 10:16
閱讀 2616·2021-08-20 09:38
閱讀 2943·2019-08-30 15:55
閱讀 1462·2019-08-30 13:21
閱讀 894·2019-08-29 15:37
閱讀 1446·2019-08-27 10:56
閱讀 2097·2019-08-26 13:45