實際業務中對於定時任務的需求是不可避免的,例如,訂單超時自動取消、天天定時拉取數據等,在Node.js中系統層面提供了setTimeout、setInterval兩個API或經過node-schedule這種第三方庫來實現。經過這種方式實現對於簡單的定時任務是ok的,過於複雜的、可用性要求較高的系統就會存在如下缺點。html
存在的一些問題node
RabbitMQ自己是不支持的,能夠經過它提供的兩個特性Time-To-Live and Expiration、Dead Letter Exchanges來實現,經過如下泳道圖能夠看到一個消息從發佈到消費的整個過程。git
死信隊列全稱 Dead-Letter-Exchange 簡稱 DLX 是 RabbitMQ 中交換器的一種類型,消息在一段時間以後沒有被消費就會變成死信被從新 publish 到另外一個 DLX 交換器隊列中,所以稱爲死信隊列。github
死信隊列產生幾種狀況shell
設置DLX的兩個參數:npm
deadLetterExchange
: 設置DLX,當正常隊列的消息成爲死信後會被路由到DLX中deadLetterRoutingKey
: 設置DLX指定的路由鍵注意
:Dead-Letter-Exchange也是一種普通的Exchangeasync
消息的TTL指的是消息的存活時間,RabbitMQ支持消息、隊列兩種方式設置TTL,分別以下:ui
x-message-ttl
或 expiration
字段設置,單位爲毫秒,表明消息的過時時間,每條消息的TTL可不一樣。x-expires
設置,隊列中的全部消息都有相同的過時時間,當超過了隊列的超時設置,消息會自動的清除。注意
:若是以上兩種方式都作了設置,消息的TTL則以二者之中最小的那個爲準。spa
推薦採用 amqplib庫,一個Node.js實現的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 隊列,造成一個死信隊列。
做者:五月君
連接:https://www.imooc.com/article...
來源:慕課網
Github: Node.js技術棧