在web開發(fā)中,我們通常需要使用定時器功能,使用setTimeout和setInterval函數。
創(chuàng)新互聯公司長期為上1000+客戶提供的網站建設服務,團隊從業(yè)經驗10年,關注不同地域、不同群體,并針對不同對象提供差異化的產品和服務;打造開放共贏平臺,與合作伙伴共同營造健康的互聯網生態(tài)環(huán)境。為北京企業(yè)提供專業(yè)的做網站、成都網站制作,北京網站改版等技術服務。擁有10余年豐富建站經驗和眾多成功案例,為您定制開發(fā)。
那么在ReactNative中,是否也提供了定時器的功能呢? 答案是肯定的。
我們還是先看看官網怎么說的。
定時器是一個應用中非常重要的部分。React Native實現了和瀏覽器一致的定時器Timer。
提供的方法如下:
setTimeout (fn, 1000) 和 setInterval (fn,1000)
和web中的意思一樣,前者表示延遲1000毫秒后執(zhí)行 fn 方法 ,后者表示每隔1000毫秒執(zhí)行 fn 方法。
requestAnimationFrame(fn)和setTimeout(fn, 0)不同,前者會在每幀刷新之后執(zhí)行一次,而后者則會盡可能快的執(zhí)行(在iPhone5S上有可能每秒1000次以上)。
setImmediate則會在當前JavaScript執(zhí)行塊結束的時候執(zhí)行,就在將要發(fā)送批量響應數據到原生之前。注意如果你在setImmediate的回調函數中又執(zhí)行了setImmediate,它會緊接著立刻執(zhí)行,而不會在調用之前等待原生代碼。
Promise的實現就使用了setImmediate來執(zhí)行異步調用。
InteractionManager(交互管理器)
原生應用感覺如此流暢的一個重要原因就是在互動和動畫的過程中避免繁重的操作。在React Native里,我們目前受到限制,因為我們只有一個JavaScript執(zhí)行線程。不過你可以用InteractionManager來確保在執(zhí)行繁重工作之前所有的交互和動畫都已經處理完畢。
應用可以通過以下代碼來安排一個任務,使其在交互結束之后執(zhí)行:
InteractionManager.runAfterInteractions(() => { // ...需要長時間同步執(zhí)行的任務... });
我們來把它和之前的幾個任務安排方法對比一下:
requestAnimationFrame(): 用來執(zhí)行在一段時間內控制視圖動畫的代碼
setImmediate/setTimeout/setInterval(): 在稍后執(zhí)行代碼。注意這有可能會延遲當前正在進行的動畫。
runAfterInteractions(): 在稍后執(zhí)行代碼,不會延遲當前進行的動畫。
觸摸處理系統會把一個或多個進行中的觸摸操作認定為'交互',并且會將runAfterInteractions()的回調函數延遲執(zhí)行,直到所有的觸摸操作都結束或取消了。
InteractionManager還允許應用注冊動畫,在動畫開始時創(chuàng)建一個交互“句柄”,然后在結束的時候清除它。
var handle = InteractionManager.createInteractionHandle(); // 執(zhí)行動畫... (`runAfterInteractions`中的任務現在開始排隊等候) // 在動畫完成之后 InteractionManager.clearInteractionHandle(handle); // 在所有句柄都清除之后,現在開始依序執(zhí)行隊列中的任務
TimerMixin
我們發(fā)現很多React Native應用發(fā)生致命錯誤(閃退)是與計時器有關。具體來說,是在某個組件被卸載(unmount)之后,計時器卻仍然被激活。為了解決這個問題,我們引入了TimerMixin。如果你在組件中引入TimerMixin,就可以把你原本的setTimeout(fn, 500)改為this.setTimeout(fn, 500)(只需要在前面加上this.),然后當你的組件卸載時,所有的計時器事件也會被正確的清除。
這個庫并沒有跟著React Native一起發(fā)布。你需要在項目文件夾下輸入npm i react-timer-mixin --save來單獨安裝它。
var TimerMixin = require('react-timer-mixin'); var Component = React.createClass({ mixins: [TimerMixin], componentDidMount: function() { this.setTimeout( () => { console.log('這樣我就不會導致內存泄露!'); }, 500 ); } });
我們強烈建議您使用react-timer-mixin提供的this.setTimeout(...)來代替setTimeout(...)。這可以規(guī)避許多難以排查的BUG。
譯注:Mixin屬于ES5語法,對于ES6代碼來說,無法直接使用Mixin。
如果你的項目是用ES6代碼編寫,同時又使用了計時器,那么你只需銘記在unmount組件時清除(clearTimeout/clearInterval)所有用到的定時器。
那么也可以實現和TimerMixin同樣的效果。例如:
import React,{ Component } from 'react-native'; export default class Hello extends Component { componentDidMount() { this.timer = setTimeout( () => { console.log('把一個定時器的引用掛在this上'); }, 500 ); } componentWillUnmount() { // 如果存在this.timer,則使用clearTimeout清空。 // 如果你使用多個timer,那么用多個變量,或者用個數組來保存引用,然后逐個clear this.timer && clearTimeout(this.timer); } };
注意點:
1、定時器功能比較簡單,注意在es6中使用時,需銘記在unmount組件時清除(clearTimeout/clearInterval)所有用到的定時器。
2、可以使用定時器實現一些普通功能:如短信倒計時等
3、對于一些需要延遲執(zhí)行的特殊場景也可以使用Timer,譬如:目前RN提供的fetch是沒有提供設置超時時間的,如果客戶端請求后端的一個接口,接口超時了(后端服務設置的超時時間為10s),那么RN界面就一直loading,也不能aborded。那么這時候我們就可以巧妙的使用計時器,如果客戶端發(fā)出的Request,時間大于某個值(5秒),那么我們直接認為請求失敗。
4、今天還發(fā)現一個使用setTimeout的場景,在列表頁加載下一頁的時候,如果接口響應很快,就不會出現loading的效果,這個時候為了有l(wèi)oading的效果,設置一個500毫秒的延時,呵呵....
參考:http://reactnative.cn/docs/0.31/timers.html#content
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持創(chuàng)新互聯。