世界上最伟大的投资就是投资自己的教育

首页PostgreSQL
随风 · 练气

PostgreSQL 的 listen/notify 之消息队列 (十三)

随风发布于4987 次阅读

1. 消息队列的简介

什么是消息队列呢?队列就是排队,就像在银行办理业务排队一样,排在最前面的先处理,后面的后处理,按照顺序来,先进先出。这个队列可以是程序,可以是数据,也可以是任务,是任何你可以存储的东西。消息队列就是给队列传递消息。这么来说,打个比方,我们在一个网站上注册了账号,系统可能会给你发送一封注册邮件,同时在页面上提示你"稍等几分钟后会收到一封邮件",发邮件这个事是通过操作系统的调用,例如 linux 的 sendmail,或者接口来发送的,发邮件是通过排队来发的,先到的先发,假如很多邮件等着发,那就得像银行那样排队了,所以未必就能实时,总有延迟。总结来说,发邮件这个事是有延迟的,是需要等待之后用户才能收到邮件的。然而,这种延迟对用户的体验还有操作并不影响啊。在网站上的其他应用他还是照样用,没有任何影响。对这种对用户没直接影响或者有延迟的任务就可以扔到消息队列处理。所以,发邮件,发短信,捕获异常等任务都可以扔到消息队列。也就是消息队列是独立于 web 进程的另一个进程,因为它有可能耗时很长的,所以要另开一个进程来处理,对 web 进程没有任务影响,用户还是照常访问网站。这么来说,假如网站有一个需要扔的消息队列叫 A,但用户触发了 A,就把 A 扔到消息队列,这时给用户感觉是这个 A 任务是一瞬间完成,其实它是给消息队列那个进程发送了消息,可能跟它说,我要发短信,就把发短信这个指令,加上短信的内容一并传给消息队列的进程,消息队列收到消息后,就把这个任务放到队列中进行排队,因为前面还有一堆任务没处理,所以要慢慢处理,轮到 A 的时候才处理 A,由于 A 是耗时的动作所以就慢慢处理就好,反正对用户不太影响。

前面说到,消息队列的进程要把任务放入队列中,由于有很多任务,需要排队,所以这些任务是需要存储起来的。在 ruby 中,有很多 gem 可以实现后台的消息队列,但它们的存储方式有区别,比如delayed_job就是用数据库 (MySQL,Sqlite,PostgreSQL 等) 来存的,它会先让你创建表,如果有任务进来,就会插入到表中作为一条记录,要处理的时候就会取出这条记录。像resquesidekiq就是用 redis 来存储数据的,redis 是存储在计算机的内存中的。比较一下,就知道 resque 和 sidekiq 在存储方式上比 delayed_job 有优势,而 delayed_job 的好处是能直接利用数据库,不用额外安装 redis。

消息队列是另外的一个进程,任务进入消息队列中,一个接一个的处理,也就是说,A 进程在被处理时,必须等前面的任务被处理完才能轮到它。这种方式体现在 delayed_job 和 resque 中。sidekiq 的处理方式是多线程的,它是基于 celluloid 的,用 Actor 作为并发模型,它能同时处理多个任务。

值得一提的是 Ruby on Rails 从 4.2 开始加上了active_job。因为有各种各样的消息队列的解决方案,active_job 就是提供了统一的接口和调用,要用到消息队列还是会用到上面提到的几个。这个东西就像 activerecord 一样,要指定数据库那也是很简单的,只要换相应的 gem 和改配置文件就行了,而 active_job 也正是这样。

不过,这一篇文章不会详说上面的三种消息队列的实现,只会说到特用于 PostgreSQL 的消息队列queue_classic

2. PostgreSQL 的 listen/notify

queue_classic是基于 PostgreSQL 的 listen/notify 来实现的,列队在等任务进来就是用的 listen,把任务放入队列就是 notify。

PostgreSQL 的listen/notify,也就是一种消息的订阅/发布模式,也就是类似那种生产者/消费者模式。这种模式很常见,例如 redis 的pub/sub模式、rails 的Notifications组件。

懂了 PostgreSQL 的 listen/notify,也就等于懂了其他的订阅/发布模式。

它很简单,就相当于一种广播机制,比如,你定阅杂志,还有其他人也订阅了,这个过程就叫 listen,也就是监听,等杂志有更新了,或者有新的杂志出来,它就会广播,就会送一份给订阅杂志的人,这个过程就是 notify,也就是通知。

listen/notify 的使用很简单。

首先是 listen(监听),只接监听的通道的名称,这个名称自己定义。

rails365_pro=# LISTEN virtual;
LISTEN

这个时候可以直接执行 notify。

rails365_pro=# NOTIFY virtual;
NOTIFY
Asynchronous notification "virtual" received from server process with PID 4996.

表示监听的通道已知收到消息了。

还可以传参数。

rails365_pro=# NOTIFY virtual, 'This is the payload';
NOTIFY
Asynchronous notification "virtual" with payload "This is the payload" received from server process with PID 4996.

也可以结合 sql 语句来使用。

LISTEN foo;
SELECT pg_notify('fo' || 'o', 'pay' || 'load');

只要连接到同一个数据库的所有 session 都会接到监听通道传过来的信息。

可以尝试另开一个 psql 进程。然后 notify,再回到之前的 psql 执行 listen 就可以测试的,如果显示正确的 pid 就成功的。

下一章: PostgreSQL 的 listen/notify 之 queue_classic(二)

完结。

本站文章均为原创内容,如需转载请注明出处,谢谢。

0 条回复
暂无回复~~
相关小书
postgresql教程

postgresql教程

postgresql 最全面,最细致的特性介绍与应用教程

发表于

喜欢
统计信息
    学员: 29746
    视频数量: 1987
    文章数量: 526

© 汕尾市求知科技有限公司 | Rails365 Gitlab | 知乎 | b 站 | csdn

粤公网安备 44152102000088号粤公网安备 44152102000088号 | 粤ICP备19038915号

Top