NodeJs之定時器與隊列
一,介紹與需求
1.1,介紹
定時任務(node-schedule),是針對Node.js的一種靈活的cron-like和not-cron-like作業調度程序。它允許您使用可選的遞歸規則將作業(任意函數)安排在特定日期執行。它在任何給定的時間只使用一個計時器(而不是每秒鍾/分鍾重新評估即將到來的作業)。
Async是一個實用模塊,它為異步JavaScript提供了直接、強大的功能。async流程控制器--queue(隊列),queue流程控制器是一個並行的流程控制器,但是它與parallel的區別在於queue可以控制一次執行幾個函數,而parallel只是讓所有函數並行執行.
1.2,需求
實際開發項目中,會遇到很多定時任務的工作。比如:定時導出某些數據、定時發送消息或郵件給用戶、定時備份什么類型的文件等等。在當時給用戶發送消息時,可能要發送的用戶就不只有一兩個,二是多個,這是可能就會用到隊列順序執行。
二,定時器
第一步:安裝node-schedule
1 npm install node-schedule --save
第二步:封裝定時器模塊
1 const schedule = require('node-schedule');//定時器 2 const nodeTimer = {}; 3 let cancelTimer = '' 4 /** 5 *Cron風格定時器/對象文本語法定時器 6 * @param executionTime :定時器字符串'30 * * * * *'/定時器對象{hour: 16, minute: 11, dayOfWeek: 1} 7 * @param callback :回調函數 8 */ 9 nodeTimer.scheduleTimer = (executionTime = '30 * * * * *', callback) => { 10 // 每分鍾的第30秒觸發: '30 * * * * *' 11 // 每小時的1分30秒觸發 :'30 1 * * * *' 12 // 每天的凌晨1點1分30秒觸發 :'30 1 1 * * *' 13 // 每月的1日1點1分30秒觸發 :'30 1 1 1 * *' 14 // 2016年的1月1日1點1分30秒觸發 :'30 1 1 1 2016 *' 15 // 每周1的1點1分30秒觸發 :'30 1 1 * * 1' 16 17 cancelTimer = schedule.scheduleJob(executionTime, () => { 18 if (typeof callback === 'function') { 19 callback() 20 } 21 }); 22 23 } 24 module.exports = nodeTimer;
第三步:調用
在回調函數中寫入要執行的任務代碼,一個定時器就完成了!
引入定時器模塊:
1 const nodeTimer = require('./node_timer.js');
1,Cron風格定時器
規則參數講解 *代表通配符
* * * * * * ┬ ┬ ┬ ┬ ┬ ┬ │ │ │ │ │ │ │ │ │ │ │ └ day of week (0 - 7) (0 or 7 is Sun) │ │ │ │ └───── month (1 - 12) │ │ │ └────────── day of month (1 - 31) │ │ └─────────────── hour (0 - 23) │ └──────────────────── minute (0 - 59) └───────────────────────── second (0 - 59, OPTIONAL)
6個占位符從左到右分別代表:秒、分、時、日、月、周幾
*
表示通配符,匹配任意,當秒是*
時,表示任意秒數都觸發,其它類推
1 // 每分鍾的第30秒觸發: '30 * * * * *' 2 // 每小時的1分30秒觸發 :'30 1 * * * *' 3 // 每天的凌晨1點1分30秒觸發 :'30 1 1 * * *' 4 // 每月的1日1點1分30秒觸發 :'30 1 1 1 * *' 5 // 2016年的1月1日1點1分30秒觸發 :'30 1 1 1 2016 *' 6 // 每周1的1點1分30秒觸發 :'30 1 1 * * 1' 7 // 每分鍾的1-10秒都會觸發,其它通配符依次類推 :'1-10 * * * * *'
調用定時器:
1 nodeTimer.scheduleTimer('30 * * * * *',function(err){ 2 if(!err){ 3 console.log('scheduleTimer:' + new Date()); 4 } 5 });
效果:
2、對象文本語法定時器
-
-
second (0-59)
minute (0-59)
hour (0-23)
date (1-31)
month (0-11)
year
dayOfWeek (0-6) Starting with Sunday
-
1 //每周一的下午15:03:30觸發,其它組合可以根據我代碼中的注釋參數名自由組合 2 nodeTimer.scheduleTimer({hour: 15, minute: 3, second: 30},function(err){ 3 if(!err){ 4 console.log('scheduleTimer:' + new Date()); 5 } 6 });
效果:
3、基於日期的定時器
1 var date = new Date(2019, 01, 07, 15, 03, 30); 2 nodeTimer.scheduleTimer(date,function(err){ 3 if(!err){ 4 console.log('scheduleTimer:' + new Date()); 5 } 6 });
4、遞歸規則定時器
參數與對象文本語法定時器的參數類似
1 var rule = new schedule.RecurrenceRule(); 2 rule.dayOfWeek = [0, new schedule.Range(4, 6)];//每周四,周五,周六執行 3 rule.hour = 15; 4 rule.minute = 0; 5 nodeTimer.scheduleTimer(rule,function(err){ 6 if(!err){ 7 console.log('scheduleTimer:' + new Date()); 8 } 9 });
5、取消定時器
1 // 取消定時器 2 // 調用 定時器對象的cancl()方法即可 3 nodeTimer.scheduleCancel = () => { 4 // 定時器取消 5 cancelTimer.cancel(); 6 console.log('定時器成功取消'); 7 }
調用:
1 nodeTimer.scheduleCancel()
效果:
三,隊列
第一步:安裝async
1 npm install --save async
第二步:封裝方法
queue相當於一個加強版的parallel,主要是限制了worker數量,不再一次性全部執行。當worker數量不夠用時,新加入的任務將會排隊等候,直到有新的worker可用。
該函數有多個點可供回調,如worker用完時、無等候任務時、全部執行完時等。
1 const async = require('async'); 2 /** 3 *隊列 4 * @param obj :obj對象 包含執行時間 5 * @param callback :回調函數 6 */ 7 const nodeQueue = async.queue(function (obj, callback) { 8 setTimeout(function () { 9 // 需要執行的代碼的回調函數 10 if(typeof callback==='function'){ 11 callback(); 12 } 13 }, obj.time) 14 }, 1) 15 16 // worker數量將用完時,會調用saturated函數 17 nodeQueue.saturated = function() { 18 console.log('all workers to be used'); 19 } 20 21 // 當最后一個任務交給worker執行時,會調用empty函數 22 nodeQueue.empty = function() { 23 console.log('no more tasks wating'); 24 } 25 26 // 當所有任務都執行完時,會調用drain函數 27 nodeQueue.drain = function() { 28 console.log('all tasks have been processed'); 29 } 30 module.exports = nodeQueue;
第三步:調用方法
1 const nodeQueue = require('./node_queue.js'); 2 for (let i = 0; i < 10; i++) { 3 nodeQueue.push({ name: 1, time: 2000 }, function (err) { 4 console.log('隊列執行錯誤信息==',err); 5 if(!err){ 6 // 需要執行的代碼或函數 7 console.log('需要執行的代碼或函數第',i+1,'個') 8 } 9 }) 10 };
效果: