5.1 Work queues 工作队列模式

1. 模式说明
image.png

  • Work Queues:与入门程序的简单模式相比,多了一个或一些消费端,多个消费端共同消费同一个队列中的消息。

  • 应用场景:对于任务过重或任务较多情况使用工作队列可以提高任务处理的速度。

    2. 代码编写

    Work Queues 与入门程序的简单模式的代码几乎是一样的。可以完全复制,并多复制一个消费者进行多 个消费者同时对消费消息的测试。

    1) 生产者代码编写 ```java /**

    • 发送消息 */ public class Producer_WorkQueues { public static void main(String[] args) throws IOException, TimeoutException {

      //1.创建连接工厂 ConnectionFactory factory = new ConnectionFactory(); //2. 设置参数 factory.setHost(“172.16.98.133”);//ip 默认值 localhost factory.setPort(5672); //端口 默认值 5672 factory.setVirtualHost(“/itcast”);//虚拟机 默认值/ factory.setUsername(“heima”);//用户名 默认 guest factory.setPassword(“heima”);//密码 默认值 guest //3. 创建连接 Connection Connection connection = factory.newConnection(); //4. 创建Channel Channel channel = connection.createChannel(); //5. 创建队列Queue /* queueDeclare(String queue, boolean durable, boolean exclusive, boolean autoDelete, Map arguments) 参数:

      1. 1. queue:队列名称
      2. 2. durable:是否持久化,当mq重启之后,还在
      3. 3. exclusive
      4. * 是否独占。只能有一个消费者监听这队列
      5. * Connection关闭时,是否删除队列
      6. *
      7. 4. autoDelete:是否自动删除。当没有Consumer时,自动删除掉
      8. 5. arguments:参数。
      9. */

      //如果没有一个名字叫hello_world的队列,则会创建该队列,如果有则不会创建 channel.queueDeclare(“work_queues”,true,false,false,null); /* basicPublish(String exchange, String routingKey, BasicProperties props, byte[] body) 参数:

      1. 1. exchange:交换机名称。简单模式下交换机会使用默认的 ""
      2. 2. routingKey:路由名称
      3. 3. props:配置信息
      4. 4. body:发送消息数据
      5. */

      for (int i = 1; i <= 10; i++) {

      1. String body = i+"hello rabbitmq~~~";
      2. //6. 发送消息
      3. channel.basicPublish("","work_queues",null,body.getBytes());

      }

  1. //7.释放资源
  2. channel.close();
  3. connection.close();
  4. }

}

  1. **2)消费者代码编写**
  2. Consumer_WorkQueues1.java
  3. ```java
  4. public class Consumer_WorkQueues1 {
  5. public static void main(String[] args) throws IOException, TimeoutException {
  6. //1.创建连接工厂
  7. ConnectionFactory factory = new ConnectionFactory();
  8. //2. 设置参数
  9. factory.setHost("172.16.98.133");//ip 默认值 localhost
  10. factory.setPort(5672); //端口 默认值 5672
  11. factory.setVirtualHost("/itcast");//虚拟机 默认值/
  12. factory.setUsername("heima");//用户名 默认 guest
  13. factory.setPassword("heima");//密码 默认值 guest
  14. //3. 创建连接 Connection
  15. Connection connection = factory.newConnection();
  16. //4. 创建Channel
  17. Channel channel = connection.createChannel();
  18. /*
  19. basicConsume(String queue, boolean autoAck, Consumer callback)
  20. 参数:
  21. 1. queue:队列名称
  22. 2. autoAck:是否自动确认
  23. 3. callback:回调对象
  24. */
  25. // 接收消息
  26. Consumer consumer = new DefaultConsumer(channel){
  27. /*
  28. 回调方法,当收到消息后,会自动执行该方法
  29. 1. consumerTag:标识
  30. 2. envelope:获取一些信息,交换机,路由key...
  31. 3. properties:配置信息
  32. 4. body:数据
  33. */
  34. @Override
  35. public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
  36. /* System.out.println("consumerTag:"+consumerTag);
  37. System.out.println("Exchange:"+envelope.getExchange());
  38. System.out.println("RoutingKey:"+envelope.getRoutingKey());
  39. System.out.println("properties:"+properties);*/
  40. System.out.println("body:"+new String(body));
  41. }
  42. };
  43. channel.basicConsume("work_queues",true,consumer);
  44. //关闭资源?不要
  45. }
  46. }

Consumer_WorkQueues2.java

  1. public class Consumer_WorkQueues2 {
  2. public static void main(String[] args) throws IOException, TimeoutException {
  3. //1.创建连接工厂
  4. ConnectionFactory factory = new ConnectionFactory();
  5. //2. 设置参数
  6. factory.setHost("172.16.98.133");//ip 默认值 localhost
  7. factory.setPort(5672); //端口 默认值 5672
  8. factory.setVirtualHost("/itcast");//虚拟机 默认值/
  9. factory.setUsername("heima");//用户名 默认 guest
  10. factory.setPassword("heima");//密码 默认值 guest
  11. //3. 创建连接 Connection
  12. Connection connection = factory.newConnection();
  13. //4. 创建Channel
  14. Channel channel = connection.createChannel();
  15. /*
  16. basicConsume(String queue, boolean autoAck, Consumer callback)
  17. 参数:
  18. 1. queue:队列名称
  19. 2. autoAck:是否自动确认
  20. 3. callback:回调对象
  21. */
  22. // 接收消息
  23. Consumer consumer = new DefaultConsumer(channel){
  24. /*
  25. 回调方法,当收到消息后,会自动执行该方法
  26. 1. consumerTag:标识
  27. 2. envelope:获取一些信息,交换机,路由key...
  28. 3. properties:配置信息
  29. 4. body:数据
  30. */
  31. @Override
  32. public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
  33. /* System.out.println("consumerTag:"+consumerTag);
  34. System.out.println("Exchange:"+envelope.getExchange());
  35. System.out.println("RoutingKey:"+envelope.getRoutingKey());
  36. System.out.println("properties:"+properties);*/
  37. System.out.println("body:"+new String(body));
  38. }
  39. };
  40. channel.basicConsume("work_queues",true,consumer);
  41. //关闭资源?不要
  42. }
  43. }

3. 小结

  1. 在一个队列中如果有多个消费者,那么消费者之间对于同一个消息的关系是竞争的关系。

  2. Work Queues 对于任务过重或任务较多情况使用工作队列可以提高任务处理的速度。例如:短信服务部署多个, 只需要有一个节点成功发送即可。

5.2 Pub/Sub 订阅模式

1. 模式说明
image.png
在订阅模型中,多了一个 Exchange 角色,而且过程略有变化:

  • P:生产者,也就是要发送消息的程序,但是不再发送到队列中,而是发给X(交换机)
  • C:消费者,消息的接收者,会一直等待消息到来
  • Queue:消息队列,接收消息、缓存消息
  • Exchange:交换机(X)。一方面,接收生产者发送的消息。另一方面,知道如何处理消息,例如递交给某个特别队列、 递交给所有队列、或是将消息丢弃。到底如何操作,取决于Exchange的类型。

Exchange有常见以下3种类型:

➢ Fanout:广播,将消息交给所有绑定到交换机的队列
➢ Direct:定向,把消息交给符合指定routing key 的队列
➢ Topic:通配符,把消息交给符合routing pattern(路由模式) 的队列

Exchange(交换机)只负责转发消息,不具备存储消息的能力,因此如果没有任何队列与 Exchange 绑定,或者没有符合 路由规则的队列,那么消息会丢失!

1)生产者代码编写

  1. /**
  2. * 发送消息
  3. */
  4. public class Producer_PubSub {
  5. public static void main(String[] args) throws IOException, TimeoutException {
  6. //1.创建连接工厂
  7. ConnectionFactory factory = new ConnectionFactory();
  8. //2. 设置参数
  9. factory.setHost("172.16.98.133");//ip 默认值 localhost
  10. factory.setPort(5672); //端口 默认值 5672
  11. factory.setVirtualHost("/itcast");//虚拟机 默认值/
  12. factory.setUsername("heima");//用户名 默认 guest
  13. factory.setPassword("heima");//密码 默认值 guest
  14. //3. 创建连接 Connection
  15. Connection connection = factory.newConnection();
  16. //4. 创建Channel
  17. Channel channel = connection.createChannel();
  18. /*
  19. exchangeDeclare(String exchange, BuiltinExchangeType type, boolean durable, boolean autoDelete, boolean internal, Map<String, Object> arguments)
  20. 参数:
  21. 1. exchange:交换机名称
  22. 2. type:交换机类型
  23. DIRECT("direct"),:定向
  24. FANOUT("fanout"),:扇形(广播),发送消息到每一个与之绑定队列。
  25. TOPIC("topic"),通配符的方式
  26. HEADERS("headers");参数匹配
  27. 3. durable:是否持久化
  28. 4. autoDelete:自动删除
  29. 5. internal:内部使用。 一般false
  30. 6. arguments:参数
  31. */
  32. String exchangeName = "test_fanout";
  33. //5. 创建交换机
  34. channel.exchangeDeclare(exchangeName, BuiltinExchangeType.FANOUT,true,false,false,null);
  35. //6. 创建队列
  36. String queue1Name = "test_fanout_queue1";
  37. String queue2Name = "test_fanout_queue2";
  38. channel.queueDeclare(queue1Name,true,false,false,null);
  39. channel.queueDeclare(queue2Name,true,false,false,null);
  40. //7. 绑定队列和交换机
  41. /*
  42. queueBind(String queue, String exchange, String routingKey)
  43. 参数:
  44. 1. queue:队列名称
  45. 2. exchange:交换机名称
  46. 3. routingKey:路由键,绑定规则
  47. 如果交换机的类型为fanout ,routingKey设置为""
  48. */
  49. channel.queueBind(queue1Name,exchangeName,"");
  50. channel.queueBind(queue2Name,exchangeName,"");
  51. String body = "日志信息:张三调用了findAll方法...日志级别:info...";
  52. //8. 发送消息
  53. channel.basicPublish(exchangeName,"",null,body.getBytes());
  54. //9. 释放资源
  55. channel.close();
  56. connection.close();
  57. }
  58. }

2)消费者代码编写

Consumer_PubSub1.java

  1. public class Consumer_PubSub1 {
  2. public static void main(String[] args) throws IOException, TimeoutException {
  3. //1.创建连接工厂
  4. ConnectionFactory factory = new ConnectionFactory();
  5. //2. 设置参数
  6. factory.setHost("172.16.98.133");//ip 默认值 localhost
  7. factory.setPort(5672); //端口 默认值 5672
  8. factory.setVirtualHost("/itcast");//虚拟机 默认值/
  9. factory.setUsername("heima");//用户名 默认 guest
  10. factory.setPassword("heima");//密码 默认值 guest
  11. //3. 创建连接 Connection
  12. Connection connection = factory.newConnection();
  13. //4. 创建Channel
  14. Channel channel = connection.createChannel();
  15. String queue1Name = "test_fanout_queue1";
  16. String queue2Name = "test_fanout_queue2";
  17. /*
  18. basicConsume(String queue, boolean autoAck, Consumer callback)
  19. 参数:
  20. 1. queue:队列名称
  21. 2. autoAck:是否自动确认
  22. 3. callback:回调对象
  23. */
  24. // 接收消息
  25. Consumer consumer = new DefaultConsumer(channel){
  26. /*
  27. 回调方法,当收到消息后,会自动执行该方法
  28. 1. consumerTag:标识
  29. 2. envelope:获取一些信息,交换机,路由key...
  30. 3. properties:配置信息
  31. 4. body:数据
  32. */
  33. @Override
  34. public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
  35. /* System.out.println("consumerTag:"+consumerTag);
  36. System.out.println("Exchange:"+envelope.getExchange());
  37. System.out.println("RoutingKey:"+envelope.getRoutingKey());
  38. System.out.println("properties:"+properties);*/
  39. System.out.println("body:"+new String(body));
  40. System.out.println("将日志信息打印到控制台.....");
  41. }
  42. };
  43. channel.basicConsume(queue1Name,true,consumer);
  44. //关闭资源?不要
  45. }
  46. }

Consumer_PubSub2.java

  1. public class Consumer_PubSub2 {
  2. public static void main(String[] args) throws IOException, TimeoutException {
  3. //1.创建连接工厂
  4. ConnectionFactory factory = new ConnectionFactory();
  5. //2. 设置参数
  6. factory.setHost("172.16.98.133");//ip 默认值 localhost
  7. factory.setPort(5672); //端口 默认值 5672
  8. factory.setVirtualHost("/itcast");//虚拟机 默认值/
  9. factory.setUsername("heima");//用户名 默认 guest
  10. factory.setPassword("heima");//密码 默认值 guest
  11. //3. 创建连接 Connection
  12. Connection connection = factory.newConnection();
  13. //4. 创建Channel
  14. Channel channel = connection.createChannel();
  15. String queue1Name = "test_fanout_queue1";
  16. String queue2Name = "test_fanout_queue2";
  17. /*
  18. basicConsume(String queue, boolean autoAck, Consumer callback)
  19. 参数:
  20. 1. queue:队列名称
  21. 2. autoAck:是否自动确认
  22. 3. callback:回调对象
  23. */
  24. // 接收消息
  25. Consumer consumer = new DefaultConsumer(channel){
  26. /*
  27. 回调方法,当收到消息后,会自动执行该方法
  28. 1. consumerTag:标识
  29. 2. envelope:获取一些信息,交换机,路由key...
  30. 3. properties:配置信息
  31. 4. body:数据
  32. */
  33. @Override
  34. public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
  35. /* System.out.println("consumerTag:"+consumerTag);
  36. System.out.println("Exchange:"+envelope.getExchange());
  37. System.out.println("RoutingKey:"+envelope.getRoutingKey());
  38. System.out.println("properties:"+properties);*/
  39. System.out.println("body:"+new String(body));
  40. System.out.println("将日志信息保存数据库.....");
  41. }
  42. };
  43. channel.basicConsume(queue2Name,true,consumer);
  44. //关闭资源?不要
  45. }
  46. }

2. 小结

  1. 交换机需要与队列进行绑定,绑定之后;一个消息可以被多个消费者都收到。
  1. 发布订阅模式与工作队列模式的区别:
  • 工作队列模式不用定义交换机,而发布/订阅模式需要定义交换机 ;
  • 发布/订阅模式的生产方是面向交换机发送消息,工作队列模式的生产方是面向队列发送消息(底层使用 默认交换机) ;
  • 发布/订阅模式需要设置队列和交换机的绑定,工作队列模式不需要设置,实际上工作队列模式会将队列绑 定到默认的交换机 ;

5.3 Routing 路由模式

1.模式说明

  • 队列与交换机的绑定,不能是任意绑定了,而是要指定一个 RoutingKey(路由key)

  • 消息的发送方在向 Exchange 发送消息时,也必须指定消息的 RoutingKey

  • Exchange 不再把消息交给每一个绑定的队列,而是根据消息的 Routing Key 进行判断,只有队列的 Routingkey 与消息的 Routing key 完全一致,才会接收到消息


    image.png
    图解:

  • P:生产者,向 Exchange 发送消息,发送消息时,会指定一个routing key

  • X:Exchange(交换机),接收生产者的消息,然后把消息递交给与 routing key 完全匹配的队列
  • C1:消费者,其所在队列指定了需要 routing key 为 error 的消息
  • C2:消费者,其所在队列指定了需要 routing key 为 info、error、warning 的消息

    1)生产者代码编写

    1. /**
    2. * 发送消息
    3. */
    4. public class Producer_Routing {
    5. public static void main(String[] args) throws IOException, TimeoutException {
    6. //1.创建连接工厂
    7. ConnectionFactory factory = new ConnectionFactory();
    8. //2. 设置参数
    9. factory.setHost("172.16.98.133");//ip 默认值 localhost
    10. factory.setPort(5672); //端口 默认值 5672
    11. factory.setVirtualHost("/itcast");//虚拟机 默认值/
    12. factory.setUsername("heima");//用户名 默认 guest
    13. factory.setPassword("heima");//密码 默认值 guest
    14. //3. 创建连接 Connection
    15. Connection connection = factory.newConnection();
    16. //4. 创建Channel
    17. Channel channel = connection.createChannel();
    18. /*
    19. exchangeDeclare(String exchange, BuiltinExchangeType type, boolean durable, boolean autoDelete, boolean internal, Map<String, Object> arguments)
    20. 参数:
    21. 1. exchange:交换机名称
    22. 2. type:交换机类型
    23. DIRECT("direct"),:定向
    24. FANOUT("fanout"),:扇形(广播),发送消息到每一个与之绑定队列。
    25. TOPIC("topic"),通配符的方式
    26. HEADERS("headers");参数匹配
    27. 3. durable:是否持久化
    28. 4. autoDelete:自动删除
    29. 5. internal:内部使用。 一般false
    30. 6. arguments:参数
    31. */
    32. String exchangeName = "test_direct";
    33. //5. 创建交换机
    34. channel.exchangeDeclare(exchangeName, BuiltinExchangeType.DIRECT,true,false,false,null);
    35. //6. 创建队列
    36. String queue1Name = "test_direct_queue1";
    37. String queue2Name = "test_direct_queue2";
    38. channel.queueDeclare(queue1Name,true,false,false,null);
    39. channel.queueDeclare(queue2Name,true,false,false,null);
    40. //7. 绑定队列和交换机
    41. /*
    42. queueBind(String queue, String exchange, String routingKey)
    43. 参数:
    44. 1. queue:队列名称
    45. 2. exchange:交换机名称
    46. 3. routingKey:路由键,绑定规则
    47. 如果交换机的类型为fanout ,routingKey设置为""
    48. */
    49. //队列1绑定 error
    50. channel.queueBind(queue1Name,exchangeName,"error");
    51. //队列2绑定 info error warning
    52. channel.queueBind(queue2Name,exchangeName,"info");
    53. channel.queueBind(queue2Name,exchangeName,"error");
    54. channel.queueBind(queue2Name,exchangeName,"warning");
    55. String body = "日志信息:张三调用了delete方法...出错误了。。。日志级别:error...";
    56. //8. 发送消息
    57. channel.basicPublish(exchangeName,"warning",null,body.getBytes());
    58. //9. 释放资源
    59. channel.close();
    60. connection.close();
    61. }
    62. }

    2)消费者代码编写

Consumer_Routing1.java

public class Consumer_Routing1 {
    public static void main(String[] args) throws IOException, TimeoutException {

        //1.创建连接工厂
        ConnectionFactory factory = new ConnectionFactory();
        //2. 设置参数
        factory.setHost("172.16.98.133");//ip  默认值 localhost
        factory.setPort(5672); //端口  默认值 5672
        factory.setVirtualHost("/itcast");//虚拟机 默认值/
        factory.setUsername("heima");//用户名 默认 guest
        factory.setPassword("heima");//密码 默认值 guest
        //3. 创建连接 Connection
        Connection connection = factory.newConnection();
        //4. 创建Channel
        Channel channel = connection.createChannel();


        String queue1Name = "test_direct_queue1";
        String queue2Name = "test_direct_queue2";


        /*
        basicConsume(String queue, boolean autoAck, Consumer callback)
        参数:
            1. queue:队列名称
            2. autoAck:是否自动确认
            3. callback:回调对象

         */
        // 接收消息
        Consumer consumer = new DefaultConsumer(channel){
            /*
                回调方法,当收到消息后,会自动执行该方法

                1. consumerTag:标识
                2. envelope:获取一些信息,交换机,路由key...
                3. properties:配置信息
                4. body:数据

             */
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
              /*  System.out.println("consumerTag:"+consumerTag);
                System.out.println("Exchange:"+envelope.getExchange());
                System.out.println("RoutingKey:"+envelope.getRoutingKey());
                System.out.println("properties:"+properties);*/
                System.out.println("body:"+new String(body));
                System.out.println("将日志信息打印到控制台.....");
            }
        };
        channel.basicConsume(queue2Name,true,consumer);


        //关闭资源?不要

    }
}

Consumer_Routing2.java

public class Consumer_Routing2 {
    public static void main(String[] args) throws IOException, TimeoutException {

        //1.创建连接工厂
        ConnectionFactory factory = new ConnectionFactory();
        //2. 设置参数
        factory.setHost("172.16.98.133");//ip  默认值 localhost
        factory.setPort(5672); //端口  默认值 5672
        factory.setVirtualHost("/itcast");//虚拟机 默认值/
        factory.setUsername("heima");//用户名 默认 guest
        factory.setPassword("heima");//密码 默认值 guest
        //3. 创建连接 Connection
        Connection connection = factory.newConnection();
        //4. 创建Channel
        Channel channel = connection.createChannel();


        String queue1Name = "test_direct_queue1";
        String queue2Name = "test_direct_queue2";


        /*
        basicConsume(String queue, boolean autoAck, Consumer callback)
        参数:
            1. queue:队列名称
            2. autoAck:是否自动确认
            3. callback:回调对象

         */
        // 接收消息
        Consumer consumer = new DefaultConsumer(channel){
            /*
                回调方法,当收到消息后,会自动执行该方法

                1. consumerTag:标识
                2. envelope:获取一些信息,交换机,路由key...
                3. properties:配置信息
                4. body:数据

             */
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
              /*  System.out.println("consumerTag:"+consumerTag);
                System.out.println("Exchange:"+envelope.getExchange());
                System.out.println("RoutingKey:"+envelope.getRoutingKey());
                System.out.println("properties:"+properties);*/
                System.out.println("body:"+new String(body));
                System.out.println("将日志信息存储到数据库.....");
            }
        };
        channel.basicConsume(queue1Name,true,consumer);


        //关闭资源?不要

    }
}

2. 小结

Routing 模式要求队列在绑定交换机时要指定 routing key,消息会转发到符合 routing key 的队列。

5.4 Topics 通配符模式

1. 模式说明

  • Topic 类型与 Direct 相比,都是可以根据 RoutingKey 把消息路由到不同的队列。只不过 Topic 类型 Exchange 可以让队列在绑定 Routing key 的时候使用通配符!

  • Routingkey 一般都是有一个或多个单词组成,多个单词之间以”.”分割,例如: item.insert

  • 通配符规则:# 匹配一个或多个词* 匹配不多不少恰好1个词,例如:item.# 能够匹配 item.insert.abc 或者 item.insert,item.* 只能匹配 item.insert

    image.png

    图解:

  • 红色 Queue:绑定的是 usa.# ,因此凡是以 usa. 开头的 routing key 都会被匹配到

  • 黄色 Queue:绑定的是 #.news ,因此凡是以 .news 结尾的 routing key 都会被匹配

1)生产者代码编写

/**
 * 发送消息
 */
public class Producer_Topics {
    public static void main(String[] args) throws IOException, TimeoutException {

        //1.创建连接工厂
        ConnectionFactory factory = new ConnectionFactory();
        //2. 设置参数
        factory.setHost("172.16.98.133");//ip  默认值 localhost
        factory.setPort(5672); //端口  默认值 5672
        factory.setVirtualHost("/itcast");//虚拟机 默认值/
        factory.setUsername("heima");//用户名 默认 guest
        factory.setPassword("heima");//密码 默认值 guest
        //3. 创建连接 Connection
        Connection connection = factory.newConnection();
        //4. 创建Channel
        Channel channel = connection.createChannel();
       /*

       exchangeDeclare(String exchange, BuiltinExchangeType type, boolean durable, boolean autoDelete, boolean internal, Map<String, Object> arguments)
       参数:
        1. exchange:交换机名称
        2. type:交换机类型
            DIRECT("direct"),:定向
            FANOUT("fanout"),:扇形(广播),发送消息到每一个与之绑定队列。
            TOPIC("topic"),通配符的方式
            HEADERS("headers");参数匹配

        3. durable:是否持久化
        4. autoDelete:自动删除
        5. internal:内部使用。 一般false
        6. arguments:参数
        */

       String exchangeName = "test_topic";
        //5. 创建交换机
        channel.exchangeDeclare(exchangeName, BuiltinExchangeType.TOPIC,true,false,false,null);
        //6. 创建队列
        String queue1Name = "test_topic_queue1";
        String queue2Name = "test_topic_queue2";
        channel.queueDeclare(queue1Name,true,false,false,null);
        channel.queueDeclare(queue2Name,true,false,false,null);
        //7. 绑定队列和交换机
        /*
        queueBind(String queue, String exchange, String routingKey)
        参数:
            1. queue:队列名称
            2. exchange:交换机名称
            3. routingKey:路由键,绑定规则
                如果交换机的类型为fanout ,routingKey设置为""
         */

        // routing key  系统的名称.日志的级别。
        //=需求: 所有error级别的日志存入数据库,所有order系统的日志存入数据库
        channel.queueBind(queue1Name,exchangeName,"#.error");
        channel.queueBind(queue1Name,exchangeName,"order.*");
        channel.queueBind(queue2Name,exchangeName,"*.*");

        String body = "日志信息:张三调用了findAll方法...日志级别:info...";
        //8. 发送消息
        channel.basicPublish(exchangeName,"goods.error",null,body.getBytes());

        //9. 释放资源
        channel.close();
        connection.close();

    }
}

2)消费者代码编写

Consumer_Topic1.java

public class Consumer_Topic1 {
    public static void main(String[] args) throws IOException, TimeoutException {

        //1.创建连接工厂
        ConnectionFactory factory = new ConnectionFactory();
        //2. 设置参数
        factory.setHost("172.16.98.133");//ip  默认值 localhost
        factory.setPort(5672); //端口  默认值 5672
        factory.setVirtualHost("/itcast");//虚拟机 默认值/
        factory.setUsername("heima");//用户名 默认 guest
        factory.setPassword("heima");//密码 默认值 guest
        //3. 创建连接 Connection
        Connection connection = factory.newConnection();
        //4. 创建Channel
        Channel channel = connection.createChannel();


        String queue1Name = "test_topic_queue1";
        String queue2Name = "test_topic_queue2";


        /*
        basicConsume(String queue, boolean autoAck, Consumer callback)
        参数:
            1. queue:队列名称
            2. autoAck:是否自动确认
            3. callback:回调对象

         */
        // 接收消息
        Consumer consumer = new DefaultConsumer(channel){
            /*
                回调方法,当收到消息后,会自动执行该方法

                1. consumerTag:标识
                2. envelope:获取一些信息,交换机,路由key...
                3. properties:配置信息
                4. body:数据

             */
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
              /*  System.out.println("consumerTag:"+consumerTag);
                System.out.println("Exchange:"+envelope.getExchange());
                System.out.println("RoutingKey:"+envelope.getRoutingKey());
                System.out.println("properties:"+properties);*/
                System.out.println("body:"+new String(body));
                System.out.println("将日志信息存入数据库.......");
            }
        };
        channel.basicConsume(queue1Name,true,consumer);


        //关闭资源?不要

    }
}

Consumer_Topic2.java

public class Consumer_Topic2 {
    public static void main(String[] args) throws IOException, TimeoutException {

        //1.创建连接工厂
        ConnectionFactory factory = new ConnectionFactory();
        //2. 设置参数
        factory.setHost("172.16.98.133");//ip  默认值 localhost
        factory.setPort(5672); //端口  默认值 5672
        factory.setVirtualHost("/itcast");//虚拟机 默认值/
        factory.setUsername("heima");//用户名 默认 guest
        factory.setPassword("heima");//密码 默认值 guest
        //3. 创建连接 Connection
        Connection connection = factory.newConnection();
        //4. 创建Channel
        Channel channel = connection.createChannel();


        String queue1Name = "test_topic_queue1";
        String queue2Name = "test_topic_queue2";


        /*
        basicConsume(String queue, boolean autoAck, Consumer callback)
        参数:
            1. queue:队列名称
            2. autoAck:是否自动确认
            3. callback:回调对象

         */
        // 接收消息
        Consumer consumer = new DefaultConsumer(channel){
            /*
                回调方法,当收到消息后,会自动执行该方法

                1. consumerTag:标识
                2. envelope:获取一些信息,交换机,路由key...
                3. properties:配置信息
                4. body:数据

             */
            @Override
            public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
              /*  System.out.println("consumerTag:"+consumerTag);
                System.out.println("Exchange:"+envelope.getExchange());
                System.out.println("RoutingKey:"+envelope.getRoutingKey());
                System.out.println("properties:"+properties);*/
                System.out.println("body:"+new String(body));
                System.out.println("将日志信息打印控制台.......");
            }
        };
        channel.basicConsume(queue2Name,true,consumer);


        //关闭资源?不要

    }
}

2. 小结

Topic 主题模式可以实现 Pub/Sub 发布与订阅模式和 Routing 路由模式的功能,只是 Topic 在配置routing key 的时候可以使用通配符,显得更加灵活。

5.5 工作模式总结

1. 简单模式 HelloWorld

一个生产者、一个消费者,不需要设置交换机(使用默认的交换机)。

2. 工作队列模式

Work Queue 一个生产者、多个消费者(竞争关系),不需要设置交换机(使用默认的交换机)。

3. 发布订阅模式 Publish/subscribe

需要设置类型为 fanout 的交换机,并且交换机和队列进行绑定,当发送消息到交换机后,交换机会将消 息发送到绑定的队列。

4. 路由模式 Routing

需要设置类型为 direct 的交换机,交换机和队列进行绑定,并且指定 routing key,当发送消息到交换机 后,交换机会根据 routing key 将消息发送到对应的队列。

5. 通配符模式 Topic

需要设置类型为 topic 的交换机,交换机和队列进行绑定,并且指定通配符方式的 routing key,当发送 消息到交换机后,交换机会根据 routing key 将消息发送到对应的队列。