实际业务中对于定时任务的需求是不可避免的,例如,订单超时自动取消、天天定时拉取数据等,在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技术栈