javascript 使用sleep函數(shù)的常見方法詳解
本文實例講述了javascript 使用sleep函數(shù)的常見方法。分享給大家供大家參考,具體如下:
一.什么是sleep函數(shù)?
花一點時間來聊一下sleep函數(shù),首先什么是sleep函數(shù)?
sleep是一種函數(shù),他的作用是使程序暫停指定的時間,起到延時的效果。
例如:
console.log(’1’);sleep(5000);console.log(’2’);
控制臺輸出數(shù)字1后會間隔5秒后輸出數(shù)字2
當(dāng)然上面的代碼是不能執(zhí)行的,因為js中是沒有sleep方法的。
所以這一篇文章主要介紹幾種在js中實現(xiàn)sleep的方式。
二.為什么使用sleep?
看到這里有人會問了,為什么要使用sleep,上面的例子我可以使用setTimeout來實現(xiàn)啊?
因為setTimeout是通過回調(diào)函數(shù)來實現(xiàn)定時任務(wù)的,所以在多任務(wù)的場景下就會出現(xiàn)回調(diào)嵌套:
console.time(’runTime:’);setTimeout(function(){ console.log(’1’) setTimeout(function(){ console.log(’2’); setTimeout(function(){ console.log(’3’); console.timeEnd(’runTime:’); }, 2000); }, 3000);}, 2000);// 1// 2// 3// runTime:: 7013.104ms
上面的方式存在回調(diào)嵌套的問題,我們希望有一個優(yōu)雅的方式來實現(xiàn)上面的例子:
sleep(2000);console.log(’1’);sleep(3000);console.log(’2’);sleep(2000);console.log(’3’);...
三.實現(xiàn)sleep
接下來我們就分別用幾種不同的方法來實現(xiàn)下sleep方法
1.基于Date實現(xiàn)
通過死循環(huán)來阻止代碼執(zhí)行,同時不停比對是否超時。
function sleep(time){ var timeStamp = new Date().getTime(); var endTime = timeStamp + time; while(true){ if (new Date().getTime() > endTime){ return; } }}console.time(’runTime:’);sleep(2000);console.log(’1’);sleep(3000);console.log(’2’);sleep(2000);console.log(’3’);console.timeEnd(’runTime:’);// 1// 2// 3// runTime:: 7004.301ms
缺點:
以上的代碼不會讓線程休眠,而是通過高負(fù)荷計算使cpu無暇處理其他任務(wù)。
這樣做的缺點是在sleep的過程中其他所有的任務(wù)都會被暫停,包括dom的渲染。
所以sleep的過程中程序會處于假死狀態(tài),并不會去執(zhí)行其他任務(wù)
2.基于Promise的sleep
為了解決ajax的回調(diào)嵌套問題,在jQuery1.8之后支持了Promise。但是單純的Promise只是將之前的縱向嵌套改為了橫向嵌套,
最終結(jié)果是下面的代碼:
function sleep(time){ return new Promise(function(resolve){ setTimeout(resolve, time); });}console.time(’runTime:’);console.log(’1’);sleep(1000).then(function(){ console.log(’2’); sleep(2000).then(function(){ console.log(’3’); console.timeEnd(’runTime:’); });});console.log(’a’);// 1// a// 2// 3// runTime:: 3013.476ms
這其實和之前的setTimeout嵌套沒什么區(qū)別,也很難看。
我們再次進(jìn)行優(yōu)化,使用ES6的Generator函數(shù)來改寫上面的例子
3.基于Generator函數(shù)的sleep
我們對sleep的執(zhí)行使用Generator函數(shù)來執(zhí)行,并且搭配co來進(jìn)行自執(zhí)行。
看代碼:
var co = require(’co’); function sleep(time){ return new Promise(function(resolve){ setTimeout(resolve, time); });} var run = function* (){ console.time(’runTime:’); console.log(’1’); yield sleep(2000); console.log(’2’); yield sleep(1000); console.log(’3’); console.timeEnd(’runTime:’);} co(run);console.log(’a’);// 1// a// 2// 3// runTime:: 3004.935ms
可以看到整體的代碼看起來不存在嵌套的關(guān)系,還是比較舒服的。
并且執(zhí)行過程不會發(fā)生假死情況,不會阻塞其他任務(wù)的執(zhí)行。
但是多了一個co執(zhí)行器的引用,所以還是有瑕疵。
當(dāng)然這不是最終版,因為ES7為我們帶來了新的解決方案。
4.基于async函數(shù)的sleep
ES7新增了async函數(shù),async函數(shù)最大的特點就是自帶執(zhí)行器,所以我們可以不借助co來實現(xiàn)sleep了
看代碼:
function sleep(time){ return new Promise((resolve) => setTimeout(resolve, time));} async function run(){ console.time(’runTime:’); console.log(’1’); await sleep(2000); console.log(’2’); await sleep(1000); console.log(’3’); console.timeEnd(’runTime:’);} run();console.log(’a’); // 1// a// 2// 3// runTime:: 3009.984ms
效果和之前的一樣。
5.使用child_process(子進(jìn)程)實現(xiàn)sleep函數(shù)
前面介紹了幾種比較簡單的sleep實現(xiàn),接下來看一個比較難的實現(xiàn)。
原理是將sleep放在子進(jìn)程中執(zhí)行,不會影響其他進(jìn)程,看代碼:
var childProcess = require(’child_process’);var nodeBin = process.argv[0]; function sleep(time) { childProcess.execFileSync(nodeBin, [’-e’, ’setTimeout(function() {}, ’ + time + ’);’]); // childProcess.spawnSync(nodeBin, [’-e’, ’setTimeout(function() {}, ’ + time + ’);’]);} console.time(’runTime:’);console.log(’1’);sleep(1000);console.log(’2’);sleep(2000);console.log(’3’);console.timeEnd(’runTime:’); // 1// 2// 3// runTime:: 3579.093ms
以上代碼,是通過childProcess對象的execFileSync或者spawnSync創(chuàng)建一個同步進(jìn)程,
在同步進(jìn)程中執(zhí)行定時器,定時器執(zhí)行完畢后回收進(jìn)程,程序繼續(xù)執(zhí)行。
6.使用npm sleep包
前面的內(nèi)容都是我們自己實現(xiàn)的,其實npm上已經(jīng)有很多相關(guān)的js包了。
我們來看看他們是怎么實現(xiàn)的,sleep
var sleep = require(’sleep’); console.log(’1’);console.time(’runTime:’);sleep.sleep(2); //休眠2秒鐘console.log(’2’);sleep.msleep(1000); //休眠1000毫秒console.log(’3’);sleep.usleep(1000000) //休眠1000000微秒 = 1秒console.log(’4’);console.timeEnd(’runTime:’); // 1// 2// 3// 4// runTime:: 4014.455ms
很強有沒有,sleep包是C++編寫,然后擴(kuò)展到Node來實現(xiàn)sleep函數(shù)也是一個不錯的選擇。
以上就是sleep的六種簡單實現(xiàn)。歡迎大家指出問題,我們一起進(jìn)步。
感興趣的朋友可以使用在線HTML/CSS/JavaScript代碼運行工具:http://tools.jb51.net/code/HtmlJsRun測試上述代碼運行效果。
更多關(guān)于JavaScript相關(guān)內(nèi)容可查看本站專題:《JavaScript常用函數(shù)技巧匯總》、《javascript面向?qū)ο笕腴T教程》、《JavaScript錯誤與調(diào)試技巧總結(jié)》、《JavaScript數(shù)據(jù)結(jié)構(gòu)與算法技巧總結(jié)》及《JavaScript數(shù)學(xué)運算用法總結(jié)》
希望本文所述對大家JavaScript程序設(shè)計有所幫助。
相關(guān)文章:
1. python爬蟲實戰(zhàn)之制作屬于自己的一個IP代理模塊2. Spring如何使用xml創(chuàng)建bean對象3. python實現(xiàn)在內(nèi)存中讀寫str和二進(jìn)制數(shù)據(jù)代碼4. python實現(xiàn)PolynomialFeatures多項式的方法5. HTML 絕對路徑與相對路徑概念詳細(xì)6. python 利用toapi庫自動生成api7. Java程序的編碼規(guī)范(6)8. Android Studio設(shè)置顏色拾色器工具Color Picker教程9. IntelliJ IDEA設(shè)置默認(rèn)瀏覽器的方法10. python實現(xiàn)讀取類別頻數(shù)數(shù)據(jù)畫水平條形圖案例
