Node.js結合RabbitMQ延遲佇列實現定時任務
實際業務中對於定時任務的需求是不可避免的,例如,訂單超時自動取消、每天定時拉取資料等,在Node.js中系統層面提供了setTimeout、setInterval兩個API或透過node-schedule這種第三方庫來實現。
透過這種方式實現對於簡單的定時任務是ok的,過於複雜的、可用性要求較高的系統就會存在以下缺點。
-
存在的一些問題
- 消耗系統記憶體,如果定時任務很多,長時間得不到釋放,將會一直佔用系統程式耗費記憶體。
- 單執行緒如何保障出現系統崩潰後之前的定時任務不受影響?多程式叢集模式下一致性的保證?
- setTimeout、setInterval會存在時間誤差,對於時間精度要求較高的是不行的。
-
RabbitMQ TTL+DLX 實現定時任務
RabbitMQ本身是不支援的,可以透過它提供的兩個特性、來實現,透過以下泳道圖可以看到一個訊息從釋出到消費的整個過程。
死信佇列
死信佇列全稱 Dead-Letter-Exchange 簡稱 DLX 是 RabbitMQ 中交換器的一種型別,訊息在一段時間之後沒有被消費就會變成死信被重新 publish 到另一個 DLX 交換器佇列中,因此稱為死信佇列。
-
死信佇列產生幾種情況
- 訊息被拒絕
- 訊息TTL過期
- 佇列達到最大長度
-
設定DLX的兩個引數:
-
deadLetterExchange
: 設定DLX,當正常佇列的訊息成為死信後會被路由到DLX中 -
deadLetterRoutingKey
: 設定DLX指定的路由鍵
-
注意
:Dead-Letter-Exchange也是一種普通的Exchange
訊息TTL
訊息的TTL指的是訊息的存活時間,RabbitMQ支援訊息、佇列兩種方式設定TTL,分別如下:
-
訊息設定TTL:對訊息的設定是在傳送時進行TTL設定,透過
x-message-ttl
或expiration
欄位設定,單位為毫秒,代表訊息的過期時間,每條訊息的TTL可不同。 -
佇列設定TTL:對佇列的設定是在訊息入佇列時計算,透過
x-expires
設定,佇列中的所有訊息都有相同的過期時間,當超過了佇列的超時設定,訊息會自動的清除。
注意
:如果以上兩種方式都做了設定,訊息的TTL則以兩者之中最小的那個為準。
Nodejs操作RabbitMQ實現延遲佇列
推薦採用 庫,一個Node.js實現的RabbitMQ客戶端。
- 初始化RabbitMQ
rabbitmq.js
// npm install amqplib
const amqp = require('amqplib');
let connection = null;
module.exports = {
connection,
init: () => amqp.connect('amqp://localhost:5672').then(conn => {
connection = conn;
console.log('rabbitmq connect success');
return connection;
})
}
- 生產者
/**
* 路由一個死信佇列
* @param { Object } connnection
*/
async function producerDLX(connnection) {
const testExchange = 'testEx';
const testQueue = 'testQu';
const testExchangeDLX = 'testExDLX';
const testRoutingKeyDLX = 'testRoutingKeyDLX';
const ch = await connnection.createChannel();
await ch.assertExchange(testExchange, 'direct', { durable: true });
const queueResult = await ch.assertQueue(testQueue, {
exclusive: false,
deadLetterExchange: testExchangeDLX,
deadLetterRoutingKey: testRoutingKeyDLX,
});
await ch.bindQueue(queueResult.queue, testExchange);
const msg = 'hello world!';
console.log('producer msg:', msg);
await ch.sendToQueue(queueResult.queue, new Buffer(msg), {
expiration: '10000'
});
ch.close();
}
- 消費者
consumer.js
const rabbitmq = require('./rabbitmq.js');
/**
* 消費一個死信佇列
* @param { Object } connnection
*/
async function consumerDLX(connnection) {
const testExchangeDLX = 'testExDLX';
const testRoutingKeyDLX = 'testRoutingKeyDLX';
const testQueueDLX = 'testQueueDLX';
const ch = await connnection.createChannel();
await ch.assertExchange(testExchangeDLX, 'direct', { durable: true });
const queueResult = await ch.assertQueue(testQueueDLX, {
exclusive: false,
});
await ch.bindQueue(queueResult.queue, testExchangeDLX, testRoutingKeyDLX);
await ch.consume(queueResult.queue, msg => {
console.log('consumer msg:', msg.content.toString());
}, { noAck: true });
}
// 消費訊息
rabbitmq.init().then(connection => consumerDLX(connection));
- 執行檢視
分別執行消費者和生產者,可以看到 producer 在44秒釋出了訊息,consumer 是在54秒接收到的訊息,實現了定時10秒種執行
$ node consumer # 執行消費者
[2019-05-07T08:45:23.099] [INFO] default - rabbitmq connect success
[2019-05-07T08:45:54.562] [INFO] default - consumer msg: hello world!
$ node producer # 執行生產者
[2019-05-07T08:45:43.973] [INFO] default - rabbitmq connect success
[2019-05-07T08:45:44.000] [INFO] default - producer msg: hello world!
- 管理控制檯檢視
testQu 佇列為我們定義的正常佇列訊息過期,會變成死信,會被路由到 testQueueDLX 佇列,形成一個死信佇列。
- 原始碼地址:
技術部落格:
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/1762/viewspace-2822841/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 基於訊息佇列(RabbitMQ)實現延遲任務佇列MQ
- C#通過rabbitmq實現定時任務(延時佇列)C#MQ佇列
- RabbitMQ實現延遲佇列MQ佇列
- RabbitMQ 實現延遲佇列MQ佇列
- 如何用RabbitMQ實現延遲佇列MQ佇列
- RabbitMQ、RocketMQ、Kafka延遲佇列實現MQKafka佇列
- Golang 實現 RabbitMQ 的延遲佇列GolangMQ佇列
- 使用RabbitMq原生實現延遲佇列MQ佇列
- RabbitMQ實戰《延遲佇列》MQ佇列
- 如何才能讓Spring Boot與RabbitMQ結合實現延遲佇列Spring BootMQ佇列
- 使用 RabbitMQ 實現延時佇列MQ佇列
- 基於rabbitmq延遲外掛實現分散式延遲任務MQ分散式
- Delayed Message 外掛實現 RabbitMQ 延遲佇列MQ佇列
- 實現簡單延遲佇列和分散式延遲佇列佇列分散式
- 延時佇列(RabbitMQ)佇列MQ
- RabbitMQ 學習筆記 -- 12 死信佇列 DLX + TTL 方式實現延遲佇列MQ筆記佇列
- 用 RabbitMQ 的死信佇列來做定時任務MQ佇列
- 詳細介紹Spring Boot + RabbitMQ實現延遲佇列Spring BootMQ佇列
- RabbitMQ:偽延時佇列MQ佇列
- Spring Boot(十四)RabbitMQ延遲佇列Spring BootMQ佇列
- RabbitMQ 延遲佇列實現訂單支付結果非同步階梯性通知MQ佇列非同步
- 一張圖帶你理解和實現RabbitMQ的延遲佇列功能MQ佇列
- 【RabbitMQ】一文帶你搞定RabbitMQ延遲佇列MQ佇列
- php+redis實現延遲佇列PHPRedis佇列
- RabbitMQ延時佇列的使用MQ佇列
- js定時任務佇列JS佇列
- RabbitMQ使用 prefetch_count優化佇列的消費,使用死信佇列和延遲佇列實現訊息的定時重試,golang版本MQ優化佇列Golang
- RabbitMQ釋出訂閱實戰-實現延時重試佇列MQ佇列
- Laravel 延遲佇列Laravel佇列
- redis 延遲佇列Redis佇列
- 你知道Redis可以實現延遲佇列嗎?Redis佇列
- 高可用延遲佇列設計與實現佇列
- laravel佇列實戰詳細記錄,含demo(任務鏈呼叫,延遲呼叫)Laravel佇列
- 面試官:RabbitMQ過期時間設定、死信佇列、延時佇列怎麼設計?面試MQ佇列
- Swoole來實現實時非同步任務佇列非同步佇列
- 延遲阻塞佇列 DelayQueue佇列
- hyperf redis延遲佇列Redis佇列
- go-zero 如何應對海量定時/延遲任務?Go