Go RabbitMQ(五)主题
时间:2019-04-14
本文章向大家介绍Go RabbitMQ(五)主题,主要包括Go RabbitMQ(五)主题使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
RabbitMQ topic
在之前我们将交换器的类型从fanout设置为direct后能够根据我们的选择获得响应的消息,虽然改良我们的消息日志系统,但是还有很多局限性,比如它不能基于多个标准进行路由
在我们的日志系统中我们可能不仅仅是依据消息的严重性进行订阅,还有可能同时基于消息的危险等级和消息来源,比如我们监听来自cron的危险错误和来自kern的所有日志。通过topic我们可以来实现以上功能
主题交换器(topic exchange)
消息如果发送到主题交换器的话不能使用任何的routing_key,它必须是由点分隔的单词列表。单词可以是任意的,但通常它们是与消息相关的一些特性
binding key必须是具有相同格式,topic交换器背后的逻辑跟direct交换器的逻辑类似,一个指定了routing key的消息将会被投递到所有使用binding key并与routing key 相匹配的队列中。binding key有两种特殊情况:
*
可以代表代替一个单词#
可以代替0个或多个单词
在本例子中,我们将发送所有描述动物的消息,这些消息将使用由三个单词(两个点)组成的routing_key发送。routing_key第一个单词描述速度,第二个表示颜色,第三个表示物种
队列Q1使用binding_key:*.orange.*
,队列Q2使用binding_key*.*.rabbit
和lazy.#
。总结如下:
- Q1队列将会接收所有orange的动物,比如
quick.orange.rabbit
,lazy.orange.elephant
,quick.orange.fox
, - Q2队列会接收所有跟rabbit相关和lazy类型的动物,比如
quick.orange.rabbit
,lazy.orange.elephant
,lazy.brown.fox
,lazy.pink.rabbit
- 像
quick.brown.fox
跟以上两个队列的routing_key都不匹配所以该消息会被丢弃 lazy.orange.male.rabbit
则只能匹配队列Q2
topic 交换器非常灵活并且可以表现为其他交换器
比如设置队列的binding_key为#,则队列会接收所有的消息不管routing_key是什么
当binding_key不使用特殊字段`*`和`#`的时候,此时topic交换器跟direct交换器一样
完整代码如下:
- emitLogsTopic.go
func main() {
conn, err := amqp.Dial("amqp://guest:guest@localhost:5672/")
failOnError(err, "Failed to connect to RabbitMQ")
defer conn.Close()
ch, err := conn.Channel()
failOnError(err, "Failed to open a channel")
defer ch.Close()
err = ch.ExchangeDeclare(
"logs_topic", // name
"topic", // type
true, // durable
false, // auto-deleted
false, // internal
false, // no-wait
nil, // arguments
)
failOnError(err, "Failed to declare an exchange")
body := bodyFrom(os.Args)
err = ch.Publish(
"logs_topic", // exchange
severityFrom(os.Args), // routing key
false, // mandatory
false, // immediate
amqp.Publishing{
ContentType: "text/plain",
Body: []byte(body),
})
failOnError(err, "Failed to publish a message")
log.Printf(" [x] Sent %s", body)
}
func bodyFrom(args []string) string {
var s string
if (len(args) < 3) || os.Args[2] == "" {
s = "hello"
} else {
s = strings.Join(args[2:], " ")
}
return s
}
func severityFrom(args []string) string {
var s string
if (len(args) < 2) || os.Args[1] == "" {
s = "anonymous.info"
} else {
s = os.Args[1]
}
return s
}
- receiveLogsTopic.go
func main() {
conn, err := amqp.Dial("amqp://guest:guest@localhost:5672/")
failOnError(err, "Failed to connect to RabbitMQ")
defer conn.Close()
ch, err := conn.Channel()
failOnError(err, "Failed to open a channel")
defer ch.Close()
err = ch.ExchangeDeclare(
"logs_topic", // name
"topic", // type
true, // durable
false, // auto-deleted
false, // internal
false, // no-wait
nil, // arguments
)
failOnError(err, "Failed to declare an exchange")
q, err := ch.QueueDeclare(
"", // name
false, // durable
false, // delete when usused
true, // exclusive
false, // no-wait
nil, // arguments
)
failOnError(err, "Failed to declare a queue")
if len(os.Args) < 2 {
log.Printf("Usage: %s [binding_key]...", os.Args[0])
os.Exit(0)
}
for _, s := range os.Args[1:] {
log.Printf("Binding queue %s to exchange %s with routing key %s",
q.Name, "logs_topic", s)
err = ch.QueueBind(
q.Name, // queue name
s, // routing key
"logs_topic", // exchange
false,
nil)
failOnError(err, "Failed to bind a queue")
}
msgs, err := ch.Consume(
q.Name, // queue
"", // consumer
true, // auto ack
false, // exclusive
false, // no local
false, // no wait
nil, // args
)
failOnError(err, "Failed to register a consumer")
forever := make(chan bool)
go func() {
for d := range msgs {
log.Printf(" [x] %s", d.Body)
}
}()
log.Printf(" [*] Waiting for logs. To exit press CTRL+C")
<-forever
}
- 接收所有的日志:
go run receiveLogsTopic.go "#"
- 接收来自kern的所有日志:
go run receiveLogsTopic.go "kern.*"
- 接收关于critical的日志:
go run receiveLogsTopic.go "*.critical"
- 创建多个binding:
go run receiveLogsTopic.go "kern.*" "*.critical"
- 启动发送消息的脚本:
go run receiveLogsTopic.go "kern.critical" "A critical kernel error"
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- [MySQL] 利用explain查看sql语句中使用的哪个索引
- php一步一步实现mysql协议(二) ——握手初始化
- php一步一步实现mysql协议(三) ——登录认证密码加密
- php一步一步实现mysql协议(四)——执行命令
- EdgeX foundry树莓派搭建与MQTT通讯
- .NET Core使用NPOI将Excel中的数据批量导入到MySQL
- RTSP协议网络摄像头网页无插件直播平台EasyNVR的云端综合管理平台EasyNVS提交基础配置报错原因分析
- 10.深入k8s:调度的优先级及抢占机制源码分析
- redis
- synchronized底层是怎么实现的?
- RSA
- Syncd - 开源自动化部署工具
- pointer-events:none禁用鼠标事件
- TiCDC 首个 GA 版本发布,特性与场景全揭秘
- Linux远程ssh执行命令expect使用及几种方法