RabbitMQ服务器安装
1、安装erlang
wget https://www.rabbitmq.com/releases/erlang/erlang-18.2-1.el6.x86_64.rpm
2、安装RabbitMQ
wget https://www.rabbitmq.com/releases/rabbitmq-server/v3.6.15/rabbitmq-server-3.6.15-1.el6.noarch.rpm
3、 常用命令
systemctl start rabbitmq-server 启动 rabbitmq stop 停止 rabbitmq-plugins list 插件命令 rabbitmq-plugins enable rabbitmq_management 安装管理插件 rabbitmq-plugins disable rabbitmq_management 卸载管理插件
4、 浏览器打开
端口号默认:15672<br/>
密码和用户名默认:guest
http://127.0.0.1:15672/#/
5、常见错误
错误提示:zsh: command not found: rabbitmq-plugins<br/>
解决办法:
第一种:export PATH=/usr/local/Cellar/rabbitmq/3.8.2/sbin/:$PATH<br/>
第二种:1: vim .bash_profile(前提是存在该文件,如果不存在,可以先创建mkdir .bash_profile,之后再执行vi编辑)<br/>
2:export PATH=/usr/local/Cellar/rabbitmq/3.8.2/sbin/sbin/:$PATH
最后:source ~/.bash_profile
## RabbitMQ核心概念
### Virtual Hosts管理
像mysql拥有数据库的概念并且可以指定用户对库和表等操作的权限。那RabbitMQ呢?RabbitMQ也有类似的权限管理。在RabbitMQ中可以虚拟消息服务器VirtualHost,每个VirtualHost相当于一个相对独立的RabbitMQ服务器,每个VirtualHost之间是相互隔离的。exchange、queue、message不能互通。 相当于mysql的db。Virtual Name一般以/开头<br/>
1、创建Virtual Hosts:
Admin->Virtual Hosts->Add a new virtual host<br/>
2、创建用户:Admin->Users->Add a user<br/>
2、对用户进行授权,点击需要授权的vhosts->Permissions->Set permission
## RabbitMQ五种模式
url格式:amqp:// 账号 密码@地址:端口号/vhost <br/>
1、Simple模式 最简单最常用的模式,一个消息只能被一个消费者消费<br/>
2、Work模式,一个消息只能被一个消费者消费
3、Publish/Subscribe订阅模式,消息被路由投递给多个队列,一个消息被多个消费者获取,生产端不允许指定消费
4、Routing路由模式,一个消息被多个消费者获取,并且消息的目标队列可以被生产者指定
5、Topic话题模式,一个消息被多个消息获取,消息的目标queue可用BindKey以通配符,(#:一个或多个词,*:一个词)的方式指定。
示例代码
package RabbitMQ import ( "fmt" "github.com/streadway/amqp" "log" ) //amqp:// 账号 密码@地址:端口号/vhost const MQURL = "amqp://imoocuser:imoocuser@127.0.0.1:5672/imooc" type RabbitMQ struct { //连接 conn *amqp.Connection //管道 channel *amqp.Channel //队列名称 QueueName string //交换机 Exchange string //key Simple模式 几乎用不到 Key string //连接信息 Mqurl string } //创建RabbitMQ结构体实例 func NewRabbitMQ(queuename string, exchange string,key string) *RabbitMQ { rabbitmq := &RabbitMQ{QueueName:queuename,Exchange:exchange,Key:key,Mqurl:MQURL} var err error //创建rabbitmq连接 rabbitmq.conn, err = amqp.Dial(rabbitmq.Mqurl) rabbitmq.failOnErr(err, "创建连接错误!") rabbitmq.channel,err = rabbitmq.conn.Channel() rabbitmq.failOnErr(err,"获取channel失败") return rabbitmq } //断开channel和connection func (r *RabbitMQ) Destory() { r.channel.Close() r.conn.Close() } //错误处理函数 func (r *RabbitMQ) failOnErr (err error,message string) { if err !=nil { log.Fatalf("%s:%s",message,err) panic(fmt.Sprintf("%s:%s",message, err)) } } //简单模式step:1。创建简单模式下RabbitMQ实例 func NewRabbitMQSimple(queueName string) * RabbitMQ { return NewRabbitMQ(queueName, "", "") } //订阅模式创建rabbitmq实例 func NewRabbitMQPubSub(exchangeName string) * RabbitMQ { //创建rabbitmq实例 rabbitmq := NewRabbitMQ("", exchangeName, "") var err error //获取connection rabbitmq.conn, err = amqp.Dial(rabbitmq.Mqurl) rabbitmq.failOnErr(err, "failed to connecct rabbitmq!") //获取channel rabbitmq.channel, err = rabbitmq.conn.Channel() rabbitmq.failOnErr(err, "failed to open a channel!") return rabbitmq } //订阅模式生成 func (r *RabbitMQ) PublishPub(message string) { //尝试创建交换机,不存在创建 err := r.channel.ExchangeDeclare( //交换机名称 r.Exchange, //交换机类型 广播类型 "fanout", //是否持久化 true, //是否字段删除 false, //true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间的绑定 false, //是否阻塞 true表示要等待服务器的响应 false, nil, ) r.failOnErr(err, "failed to declare an excha" + "nge") //2 发送消息 err = r.channel.Publish( r.Exchange, "", false, false, amqp.Publishing{ //类型 ContentType:"text/plain", //消息 Body:[]byte(message), }, ) } //订阅模式消费端代码 func (r * RabbitMQ) RecieveSub() { //尝试创建交换机,不存在创建 err := r.channel.ExchangeDeclare( //交换机名称 r.Exchange, //交换机类型 广播类型 "fanout", //是否持久化 true, //是否字段删除 false, //true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间的绑定 false, //是否阻塞 true表示要等待服务器的响应 false, nil, ) r.failOnErr(err, "failed to declare an excha" + "nge") //2试探性创建队列,创建队列 q, err := r.channel.QueueDeclare( "",//随机生产队列名称 false, false, true, false, nil, ) r.failOnErr(err, "Failed to declare a queue") //绑定队列到exchange中 err = r.channel.QueueBind( q.Name, //在pub/sub模式下,这里的key要为空 "", r.Exchange, false, nil, ) //消费消息 message, err := r.channel.Consume( q.Name, "", true, false, false, false, nil, ) forever := make(chan bool) go func() { for d := range message { log.Printf("Received a message:%s,", d.Body) } }() fmt.Println("退出请按 Ctrl+C") <- forever } //话题模式 创建RabbitMQ实例 func NewRabbitMQTopic(exchagne string, routingKey string) *RabbitMQ { //创建rabbitmq实例 rabbitmq := NewRabbitMQ("", exchagne, routingKey) var err error rabbitmq.conn, err = amqp.Dial(rabbitmq.Mqurl) rabbitmq.failOnErr(err,"failed to connect rabbingmq!") rabbitmq.channel,err = rabbitmq.conn.Channel() rabbitmq.failOnErr(err, "failed to open a channel") return rabbitmq } //话题模式发送信息 func (r * RabbitMQ) PublishTopic(message string) { //尝试创建交换机,不存在创建 err := r.channel.ExchangeDeclare( //交换机名称 r.Exchange, //交换机类型 话题模式 "topic", //是否持久化 true, //是否字段删除 false, //true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间的绑定 false, //是否阻塞 true表示要等待服务器的响应 false, nil, ) r.failOnErr(err, "topic failed to declare an excha" + "nge") //2发送信息 err = r.channel.Publish( r.Exchange, //要设置 r.Key, false, false, amqp.Publishing{ //类型 ContentType:"text/plain", //消息 Body:[]byte(message), }, ) } //话题模式接收信息 //要注意key //其中* 用于匹配一个单词,#用于匹配多个单词(可以是零个) //匹配 表示匹配imooc.* 表示匹配imooc.hello,但是imooc.hello.one需要用imooc.#才能匹配到 func (r *RabbitMQ) RecieveTopic() { //尝试创建交换机,不存在创建 err := r.channel.ExchangeDeclare( //交换机名称 r.Exchange, //交换机类型 话题模式 "topic", //是否持久化 true, //是否字段删除 false, //true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间的绑定 false, //是否阻塞 true表示要等待服务器的响应 false, nil, ) r.failOnErr(err, "failed to declare an excha" + "nge") //2试探性创建队列,创建队列 q, err := r.channel.QueueDeclare( "",//随机生产队列名称 false, false, true, false, nil, ) r.failOnErr(err, "Failed to declare a queue") //绑定队列到exchange中 err = r.channel.QueueBind( q.Name, //在pub/sub模式下,这里的key要为空 r.Key, r.Exchange, false, nil, ) //消费消息 message, err := r.channel.Consume( q.Name, "", true, false, false, false, nil, ) forever := make(chan bool) go func() { for d := range message { log.Printf("Received a message:%s,", d.Body) } }() fmt.Println("退出请按 Ctrl+C") <- forever } //路由模式 创建RabbitMQ实例 func NewRabbitMQRouting(exchagne string, routingKey string) *RabbitMQ { //创建rabbitmq实例 rabbitmq := NewRabbitMQ("", exchagne, routingKey) var err error rabbitmq.conn, err = amqp.Dial(rabbitmq.Mqurl) rabbitmq.failOnErr(err,"failed to connect rabbingmq!") rabbitmq.channel,err = rabbitmq.conn.Channel() rabbitmq.failOnErr(err, "failed to open a channel") return rabbitmq } //路由模式发送信息 func (r * RabbitMQ) PublishRouting(message string) { //尝试创建交换机,不存在创建 err := r.channel.ExchangeDeclare( //交换机名称 r.Exchange, //交换机类型 广播类型 "direct", //是否持久化 true, //是否字段删除 false, //true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间的绑定 false, //是否阻塞 true表示要等待服务器的响应 false, nil, ) r.failOnErr(err, "failed to declare an excha" + "nge") //发送信息 err = r.channel.Publish( r.Exchange, //要设置 r.Key, false, false, amqp.Publishing{ //类型 ContentType:"text/plain", //消息 Body:[]byte(message), }, ) } //路由模式接收信息 func (r *RabbitMQ) RecieveRouting() { //尝试创建交换机,不存在创建 err := r.channel.ExchangeDeclare( //交换机名称 r.Exchange, //交换机类型 广播类型 "direct", //是否持久化 true, //是否字段删除 false, //true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间的绑定 false, //是否阻塞 true表示要等待服务器的响应 false, nil, ) r.failOnErr(err, "failed to declare an excha" + "nge") //2试探性创建队列,创建队列 q, err := r.channel.QueueDeclare( "",//随机生产队列名称 false, false, true, false, nil, ) r.failOnErr(err, "Failed to declare a queue") //绑定队列到exchange中 err = r.channel.QueueBind( q.Name, //在pub/sub模式下,这里的key要为空 r.Key, r.Exchange, false, nil, ) //消费消息 message, err := r.channel.Consume( q.Name, "", true, false, false, false, nil, ) forever := make(chan bool) go func() { for d := range message { log.Printf("Received a message:%s,", d.Body) } }() fmt.Println("退出请按 Ctrl+C") <- forever } //简单模式Step:2、简单模式下生产代码 func (r *RabbitMQ) PublishSimple (message string) { //1、申请队列,如果队列存在就跳过,不存在创建 //优点:保证队列存在,消息能发送到队列中 _, err := r.channel.QueueDeclare( //队列名称 r.QueueName, //是否持久化 false, //是否为自动删除 当最后一个消费者断开连接之后,是否把消息从队列中删除 false, //是否具有排他性 true表示自己可见 其他用户不能访问 false, //是否阻塞 true表示要等待服务器的响应 false, //额外数学系 nil, ) if err != nil { fmt.Println(err) } //2.发送消息到队列中 r.channel.Publish( //默认的Exchange交换机是default,类型是direct直接类型 r.Exchange, //要赋值的队列名称 r.QueueName, //如果为true,根据exchange类型和routkey规则,如果无法找到符合条件的队列那么会把发送的消息返回给发送者 false, //如果为true,当exchange发送消息到队列后发现队列上没有绑定消费者,则会把消息还给发送者 false, //消息 amqp.Publishing{ //类型 ContentType:"text/plain", //消息 Body:[]byte(message), }) } func (r *RabbitMQ) ConsumeSimple() { //1、申请队列,如果队列存在就跳过,不存在创建 //优点:保证队列存在,消息能发送到队列中 _, err := r.channel.QueueDeclare( //队列名称 r.QueueName, //是否持久化 false, //是否为自动删除 当最后一个消费者断开连接之后,是否把消息从队列中删除 false, //是否具有排他性 false, //是否阻塞 false, //额外数学系 nil, ) if err != nil { fmt.Println(err) } //接收消息 msgs, err := r.channel.Consume( r.QueueName, //用来区分多个消费者 "", //是否自动应答 true, //是否具有排他性 false, //如果设置为true,表示不能同一个connection中发送的消息传递给这个connection中的消费者 false, //队列是否阻塞 false, nil, ) if err!=nil { fmt.Println(err) } forever := make(chan bool) //启用协程处理 go func() { for d := range msgs { //实现我们要处理的逻辑函数 log.Printf("Received a message:%s",d.Body) //fmt.Println(d.Body) } }() log.Printf("【*】warting for messages, To exit press CCTRAL+C") <- forever }
测试
//Simple模式 发送者 rabbitmq := RabbitMQ.NewRabbitMQSimple("imoocSimple") rabbitmq.PublishSimple("hello imooc!") //接收者 rabbitmq := RabbitMQ.NewRabbitMQSimple("imoocSimple") rabbitmq.ConsumeSimple() //订阅模式发送者 rabbitmq := RabbitMQ.NewRabbitMQPubSub("" + "newProduct") for i :=0; i<=100 ; i++ { rabbitmq.PublishPub("订阅模式生产第" + strconv.Itoa(i) + "条数据") fmt.Println(i) time.Sleep(1 * time.Second) } //接收者 rabbitmq := RabbitMQ.NewRabbitMQPubSub("" + "newProduct") rabbitmq.RecieveSub() //路由模式发送者 imoocOne := RabbitMQ.NewRabbitMQRouting("exImooc", "imooc_one") imoocTwo := RabbitMQ.NewRabbitMQRouting("exImooc", "imooc_two") for i :=0; i<=10; i++ { imoocOne.PublishRouting("hello imooc one!" + strconv.Itoa(i)) imoocTwo.PublishRouting("hello imooc two!" + strconv.Itoa(i)) time.Sleep(1 * time.Second) fmt.Println(i) } //接收者 rabbitmq := RabbitMQ.NewRabbitMQRouting("exImooc", "imooc_one") rabbitmq.RecieveRouting() //Topic模式发送者 imoocOne := RabbitMQ.NewRabbitMQTopic("exImoocTopic", "imooc.topic88.three") imoocTwo := RabbitMQ.NewRabbitMQTopic("exImoocTopic", "imooc.topic88.four") for i :=0; i<=10; i++ { imoocOne.PublishTopic("hello imooc topic three!" + strconv.Itoa(i)) imoocTwo.PublishTopic("hello imooc topic four!" + strconv.Itoa(i)) time.Sleep(1 * time.Second) fmt.Println(i) } //Topic接收者 rabbitmq := RabbitMQ.NewRabbitMQTopic("exImoocTopic", "#") rabbitmq.RecieveTopic()