RabbitMQ的各种模式

 前面我们已经说明了Simple模式的实现,下面我们会说明其他模式的使用和实现

 

 1、Work 工作模式

  特点:一个消息只能被一个消费者获取,也就是说一个消息只能被消费一次。其实就是多了些消费者

  

 

  使用场景: 生产消息的速度 大于 消费消费消息的速度,还句话说就是减少系统的负载

代码实现

 

 和Simple模式类似,就是多了一个mainWorkReceive.go,代码都是一样的。

  • RabbitMQWork/mainWorkPublish.go
    package main
    
    import (
    	"demo/RabbitMQ"
    	"fmt"
    	"strconv"
    	"time"
    )
    
    func main() {
    	rabbitmq := RabbitMQ.NewRabbitMQSimple("testSimple")
    
    	for i := 0; i <= 100; i++ {
    		rabbitmq.PublishSimple("Hello Test! "+strconv.Itoa(i))
    		time.Sleep(1*time.Second)
    		fmt.Println(i)
    	}
    }
    
  • RabbitMQWork/mainWorkRecieve.go
    package main
    
    import "demo/RabbitMQ"
    
    func main() {
    	rabbitmq := RabbitMQ.NewRabbitMQSimple("testSimple")
    	rabbitmq.ConsumeSimple()
    }
    

2、Publish/Subscribe 订阅模式

  消费被路由投递给多个队列,一个消息被多个消费者获取, 下图的x为交换机,通过规则去匹配到各自的对队列上,再转给消费者

 

 

 

代码实现

  •  RabbitMQ/rabbitmq.go
    package RabbitMQ
    
    import (
    	"fmt"
    	"github.com/streadway/amqp"
    	"log"
    )
    
    // 创建连接url
    const MQURL = "amqp://admin:admin@127.0.0.1:5672/test"
    
    type RabbitMQ struct {
    	conn    *amqp.Connection
    	channel *amqp.Channel
    
    	// 队列名称
    	QueueName string
    	// 交换机
    	Exchange string
    	// key
    	Key string
    	// 连接信息
    	Mqurl string
    }
    
    // 创建连接实例
    func NewRabbitMQ(queueName string, exchange string, key string) *RabbitMQ {
    	// exchange 为空会使用默认的default
    	rabbitmq := &RabbitMQ{QueueName: queueName, Exchange: exchange, Key: key, Mqurl: MQURL}
    	var err error
    	// 创建rabbitmq来连接
    	rabbitmq.conn, err = amqp.Dial(rabbitmq.Mqurl)
    	rabbitmq.failOnErr(err, "创建连接错误!")
    
    	// 创建channel
    	rabbitmq.channel, err = rabbitmq.conn.Channel()
    	rabbitmq.failOnErr(err, "获取channel失败")
    	return rabbitmq
    }
    
    // 断开连接:channel和conn
    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", message))
    	}
    }
    
    // step1: simple style 创建简单模式的实例, 只需要队列名
    func NewRabbitMQSimple(queueName string) *RabbitMQ {
    	return NewRabbitMQ(queueName, "", "")
    }
    
    // step2: 简单模式下生产
    func (r *RabbitMQ) PublishSimple(message string) {
    	// 固定用法  申请队列,如果队列不存在会自动创建,如果存在则直接使用,保证队列中能存入数据
    	_, err := r.channel.QueueDeclare(
    		r.QueueName, // 队列名
    		false,       // 控制是否持久化
    		false,       // 是否自动删除,当最后一个消费者断开连接后是否删除
    		false,       // 是否具有排他性,其他用户不可访问
    		false,       // 是否阻塞
    		nil,         //  额外属性
    	)
    	if err != nil {
    		fmt.Println(err)
    	}
    
    	// 发送消息到队列中
    	err = r.channel.Publish(
    		r.Exchange,
    		r.QueueName,
    		false, // mandatory 如果为true,会根据exchange类型和routkey规则,如果无法找到符合条件的队列那么会把消息返回给发送者
    		false, // immediate 如果为true,当exchange发送消息到队列后发现队列没有绑定消费者后会把消息返回
    		amqp.Publishing{
    			ContentType: "text/plain",
    			Body:        []byte(message),
    		})
    }
    
    // 简单模式的消费消息
    func (r *RabbitMQ) ConsumeSimple() {
    	// 固定用法  申请队列,如果队列不存在会自动创建,如果存在则直接使用,保证队列中能存入数据
    	_, 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,  // 是否自动应答, 主动的告诉mq自己已经消费完了,如果false,需要回调函数
    		false, // 是否排他性
    		false, // 如果设置为true, 表示不能将同一个connection中发送的消息传递给这个connect中的消费者
    		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)
    		}
    	}()
    
    	log.Printf("[*] Waiting for message,To exit press CTRL + C")
    	<-forever
    }
    
    // 订阅模式创建RabbitMQ实例, 需要指定exchange
    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 connect 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",   // 订阅模式下,需要设置为fanout, 广播类型
    		true,       //  是否持久化
    		false,      //  是否自动删除
    		false,      //  true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间绑定
    		false,      //   是否阻塞
    		nil,
    	)
    
    	r.failOnErr(err, "Failed to declare an exchange!")
    
    	// 发送信息
    	err = r.channel.Publish(
    		r.Exchange,
    		"",
    		false, // mandatory 如果为true,会根据exchange类型和routkey规则,如果无法找到符合条件的队列那么会把消息返回给发送者
    		false, // 和simple模式类似
    		amqp.Publishing{
    			ContentType: "text/plain",
    			Body:        []byte(message),
    		})
    }
    
    // 订阅模式消费端代码,
    /*
    	凡是消费者都需要将有exchange, queue对象 且将他们绑定到一个exchange上
    */
    func (r *RabbitMQ) RecieveSub() {
    	// 1 试探性创建交换机
    	err := r.channel.ExchangeDeclare(
    		r.Exchange, // 交换机名字
    		"fanout",   // 订阅模式下,需要设置为fanout, 广播类型
    		true,       //  是否持久化
    		false,      //  是否自动删除
    		false,      //  true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间绑定
    		false,      //   是否阻塞
    		nil,
    	)
    	r.failOnErr(err, "Failed to declare an exchange!")
    
    	// 试探性创建队列,这里注意队列名称不要写
    	q, err := r.channel.QueueDeclare(
    		"",    // 队列名这里为空 表示随机生产队列名称 因为每个消费者就需要一个队列
    		false, // 控制是否持久化
    		false, // 是否自动删除,当最后一个消费者断开连接后是否删除
    		true,  // 是否具有排他性 这里设置为true
    		false, // 是否阻塞
    		nil,   //  额外属性
    	)
    	r.failOnErr(err, "Failed to declare a queue!")
    
    	// 绑定队列到exchange中
    	err = r.channel.QueueBind(
    		q.Name,      // 这里就是上面创建的/存在的交换机随即名字
    		"",     // 在pub/sub模式下,这里的key为空
    		r.Exchange,  //  绑定到一个exchange上
    		false,
    		nil,
    	)
    
    	// 和Simple模式类似
    	messages, err := r.channel.Consume(
    		q.Name,
    		"",
    		true,
    		false,
    		false,
    		false,
    		nil,
    	)
    
    	forever := make(chan bool)
    	go func() {
    		for d := range messages {
    			log.Printf("Received a message: %s", d.Body)
    		}
    	}()
    	fmt.Println("退出按:CTRL + C \n")
    	<-forever
    
    }
    
  • RabbitMQSubPub/mainSub1.go
    package main
    
    import "demo/RabbitMQ"
    
    func main() {
    	rabbitmq := RabbitMQ.NewRabbitMQPubSub("newProduct")
    	rabbitmq.RecieveSub()
    }
    
  • RabbitMQSubPub/mainPub.go
    package main
    
    import (
    	"demo/RabbitMQ"
    	"strconv"
    	"time"
    )
    
    func main() {
    	rabbitmq := RabbitMQ.NewRabbitMQPubSub("newProduct")
    
    	for i := 0; i < 100; i ++ {
    		rabbitmq.PublishPub("订阅模式生产第" +strconv.Itoa(i) +"条数据")
    		time.Sleep(1 * time.Second)
    	}
    }
    

     

3、Routing 路由模式

  一个消息被多个消费在获取。并且消息的目标队列可以被生产者指定,按照指定规则匹配到相应的队列中; 作用就是不同的key去取不同的数据

 

 

  •  RabbitMQ/rabbitmq.go
    package RabbitMQ
    
    import (
    	"fmt"
    	"github.com/streadway/amqp"
    	"log"
    )
    
    // 创建连接url
    const MQURL = "amqp://admin:admin@127.0.0.1:5672/test"
    
    type RabbitMQ struct {
    	conn    *amqp.Connection
    	channel *amqp.Channel
    
    	// 队列名称
    	QueueName string
    	// 交换机
    	Exchange string
    	// key
    	Key string
    	// 连接信息
    	Mqurl string
    }
    
    // 创建连接实例
    func NewRabbitMQ(queueName string, exchange string, key string) *RabbitMQ {
    	// exchange 为空会使用默认的default
    	rabbitmq := &RabbitMQ{QueueName: queueName, Exchange: exchange, Key: key, Mqurl: MQURL}
    	var err error
    	// 创建rabbitmq来连接
    	rabbitmq.conn, err = amqp.Dial(rabbitmq.Mqurl)
    	rabbitmq.failOnErr(err, "创建连接错误!")
    
    	// 创建channel
    	rabbitmq.channel, err = rabbitmq.conn.Channel()
    	rabbitmq.failOnErr(err, "获取channel失败")
    	return rabbitmq
    }
    
    // 断开连接:channel和conn
    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", message))
    	}
    }
    
    // step1: simple style 创建简单模式的实例, 只需要队列名
    func NewRabbitMQSimple(queueName string) *RabbitMQ {
    	return NewRabbitMQ(queueName, "", "")
    }
    
    // step2: 简单模式下生产
    func (r *RabbitMQ) PublishSimple(message string) {
    	// 固定用法  申请队列,如果队列不存在会自动创建,如果存在则直接使用,保证队列中能存入数据
    	_, err := r.channel.QueueDeclare(
    		r.QueueName, // 队列名
    		false,       // 控制是否持久化
    		false,       // 是否自动删除,当最后一个消费者断开连接后是否删除
    		false,       // 是否具有排他性,其他用户不可访问
    		false,       // 是否阻塞
    		nil,         //  额外属性
    	)
    	if err != nil {
    		fmt.Println(err)
    	}
    
    	// 发送消息到队列中
    	err = r.channel.Publish(
    		r.Exchange,
    		r.QueueName,
    		false, // mandatory 如果为true,会根据exchange类型和routkey规则,如果无法找到符合条件的队列那么会把消息返回给发送者
    		false, // immediate 如果为true,当exchange发送消息到队列后发现队列没有绑定消费者后会把消息返回
    		amqp.Publishing{
    			ContentType: "text/plain",
    			Body:        []byte(message),
    		})
    }
    
    // 简单模式的消费消息
    func (r *RabbitMQ) ConsumeSimple() {
    	// 固定用法  申请队列,如果队列不存在会自动创建,如果存在则直接使用,保证队列中能存入数据
    	_, 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,  // 是否自动应答, 主动的告诉mq自己已经消费完了,如果false,需要回调函数
    		false, // 是否排他性
    		false, // 如果设置为true, 表示不能将同一个connection中发送的消息传递给这个connect中的消费者
    		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)
    		}
    	}()
    
    	log.Printf("[*] Waiting for message,To exit press CTRL + C")
    	<-forever
    }
    
    // 订阅模式创建RabbitMQ实例, 需要指定exchange
    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 connect 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",   // 订阅模式下,需要设置为fanout, 广播类型
    		true,       //  是否持久化
    		false,      //  是否自动删除
    		false,      //  true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间绑定
    		false,      //   是否阻塞
    		nil,
    	)
    
    	r.failOnErr(err, "Failed to declare an exchange!")
    
    	// 发送信息
    	err = r.channel.Publish(
    		r.Exchange,
    		"",
    		false, // mandatory 如果为true,会根据exchange类型和routkey规则,如果无法找到符合条件的队列那么会把消息返回给发送者
    		false, // 和simple模式类似
    		amqp.Publishing{
    			ContentType: "text/plain",
    			Body:        []byte(message),
    		})
    }
    
    // 订阅模式消费端代码,
    /*
    	凡是消费者都需要将有exchange, queue对象 且将他们绑定到一个exchange上
    */
    func (r *RabbitMQ) RecieveSub() {
    	// 1 试探性创建交换机
    	err := r.channel.ExchangeDeclare(
    		r.Exchange, // 交换机名字
    		"fanout",   // 订阅模式下,需要设置为fanout, 广播类型
    		true,       //  是否持久化
    		false,      //  是否自动删除
    		false,      //  true表示这个exchange不可以被client用来推送消息,仅用来进行exchange和exchange之间绑定
    		false,      //   是否阻塞
    		nil,
    	)
    	r.failOnErr(err, "Failed to declare an exchange!")
    
    	// 试探性创建队列,这里注意队列名称不要写
    	q, err := r.channel.QueueDeclare(
    		"",    // 队列名这里为空 表示随机生产队列名称 因为每个消费者就需要一个队列
    		false, // 控制是否持久化
    		false, // 是否自动删除,当最后一个消费者断开连接后是否删除
    		true,  // 是否具有排他性 这里设置为true
    		false, // 是否阻塞
    		nil,   //  额外属性
    	)
    	r.failOnErr(err, "Failed to declare a queue!")
    
    	// 绑定队列到exchange中
    	err = r.channel.QueueBind(
    		q.Name,      // 这里就是上面创建的/存在的交换机随即名字
    		"",     // 在pub/sub模式下,这里的key为空
    		r.Exchange,  //  绑定到一个exchange上
    		false,
    		nil,
    	)
    
    	// 和Simple模式类似
    	messages, err := r.channel.Consume(
    		q.Name,
    		"",
    		true,
    		false,
    		false,
    		false,
    		nil,
    	)
    
    	forever := make(chan bool)
    	go func() {
    		for d := range messages {
    			log.Printf("Received a message: %s", d.Body)
    		}
    	}()
    	fmt.Println("退出按:CTRL + C \n")
    	<-forever
    
    }
    
    /*
    	路由模式:需要将使用交换机上该为'direct', 然后将队列的key绑定到交换机上
    */
    // 路由模式, 创建RabbitMQ实例, routingKey 用来匹配规则
    func NewRabbitMQRouting(exchangeName string,routingKey string) *RabbitMQ {
    	// 创建RabbitMQ实例
    	rabbitmq := NewRabbitMQ("", exchangeName, routingKey)
    	var err error
    	// 获取connection
    	rabbitmq.conn, err = amqp.Dial(rabbitmq.Mqurl)
    	rabbitmq.failOnErr(err, "Failed to connect rabbitmq!")
    	// 获取channel
    	rabbitmq.channel, err = rabbitmq.conn.Channel()
    	rabbitmq.failOnErr(err, "failed to open channel")
    	return rabbitmq
    }
    
    // 路由模式发送消息
    func (r *RabbitMQ) PublishRouting(message string) {
    	// 1 尝试创建交换机
    	err := r.channel.ExchangeDeclare(
    			r.Exchange,
    			"direct",  // 改成direct
    			true,
    			false,
    			false,
    			false,
    			nil,
    		)
    
    	r.failOnErr(err, "Failed to declare an exchange!")
    
    	// 2 发送消息加上key
    	err = r.channel.Publish(
    			r.Exchange,
    			r.Key,   // 在设置key 将生产者中逃入这个key
    			false,
    			false,
    			amqp.Publishing{
    				ContentType:     "text/plain",
    				Body:            []byte(message),
    			})
    }
    
    //路由模式接受消息
    func (r *RabbitMQ) RecieveRouting() {
    	// 1 试探性创建交换机
    	err := r.channel.ExchangeDeclare(
    			r.Exchange,
    			"direct",
    			true,
    			false,
    			false,
    			false,
    			nil,
    		)
    	r.failOnErr(err, "Failed to declare an exchange!")
    
    	// 试探性创建队列,这里注意队列名称不要写
    	q, err := r.channel.QueueDeclare(
    		"",    // 队列名这里为空 表示随机生产队列名称 因为每个消费者就需要一个队列
    		false, // 控制是否持久化
    		false, // 是否自动删除,当最后一个消费者断开连接后是否删除
    		true,  // 是否具有排他性 这里设置为true
    		false, // 是否阻塞
    		nil,   //  额外属性
    	)
    	r.failOnErr(err, "Failed to declare a queue!")
    
    	// 绑定队列到exchange中
    	err = r.channel.QueueBind(
    		q.Name,      // 这里就是上面创建的/存在的交换机随即名字
    		r.Key,      // 需要设置key到交换机上
    		r.Exchange,  //  绑定到一个exchange上
    		false,
    		nil,
    	)
    
    	// 消费消息
    	messages, err := r.channel.Consume(
    		q.Name,
    		"",
    		true,
    		false,
    		false,
    		false,
    		nil,
    	)
    
    	forever := make(chan bool)
    	go func() {
    		for d := range messages {
    			log.Printf("Received a message: %s", d.Body)
    		}
    	}()
    	fmt.Println("退出按:CTRL + C \n")
    	<-forever
    }
    
  • RabbitMQRouting/publishRouting.go
    package main
    
    import (
    	"demo/RabbitMQ"
    	"fmt"
    	"strconv"
    	"time"
    )
    
    func main() {
    	pub := RabbitMQ.NewRabbitMQRouting("exTest","key1")
    	pub2 := RabbitMQ.NewRabbitMQRouting("exTest","key2")
    
    	for i:=0;i<10;i++{
    		pub.PublishRouting("Hello Test one!" +strconv.Itoa(i))
    		pub2.PublishRouting("Hello Test Two!" +strconv.Itoa(i))
    
    		time.Sleep(1*time.Second)
    		fmt.Println(i)
    	}
    }
    
  • RabbitMQRouting/receiveOne.go
    package main
    
    import "demo/RabbitMQ"
    
    func main() {
    	one := RabbitMQ.NewRabbitMQRouting("exTest","key1")
    	one.RecieveRouting()
    }
    
  • RabbitMQRouting/receiveTwo.go
    package main
    
    import "demo/RabbitMQ"
    
    func main() {
    	one := RabbitMQ.NewRabbitMQRouting("exTest","key2")
    	one.RecieveRouting()
    }
    

     

 

posted @ 2020-03-28 15:54  独角兕大王  阅读(260)  评论(0编辑  收藏  举报