Bootstrap

RabbitMQ基础组件封装—整体结构(总篇)

一、父项目 rabbit-parent

使用 idea 创建 maven 项目,命名为 rabbit-parent,作为 最外围的父项目,在其下创建四个 Module :rabbit-api、rabbit-core-producer、rabbit-common、rabbit-task,然后将父项目 rabbit-parent 的 src 目录删除,只保留 pom.xml,用于添加 依赖项,如下:

    <properties>
		<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
		<java.version>8</java.version>
        <fasterxml.uuid.version>3.1.4</fasterxml.uuid.version>
        <org.codehaus.jackson.version>1.9.13</org.codehaus.jackson.version>
        <druid.version>1.0.24</druid.version>
        <elastic-job.version>2.1.4</elastic-job.version>
		<guava.version>20.0</guava.version>
	    <commons-lang3.version>3.3.1</commons-lang3.version>
	    <commons-io.version>2.4</commons-io.version>
	    <commons-collections.version>3.2.2</commons-collections.version>
		<curator.version>2.11.0</curator.version>
		<fastjson.version>1.1.26</fastjson.version>        
	</properties>  
  	<dependencies>
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter</artifactId>
		</dependency>  	
		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-test</artifactId>
			<scope>test</scope>
		</dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <scope>provided</scope>
        </dependency>		
		<dependency>
		    <groupId>com.google.guava</groupId>
		    <artifactId>guava</artifactId>
		    <version>${guava.version}</version>
		</dependency>
		<dependency>
			<groupId>org.apache.commons</groupId>
			<artifactId>commons-lang3</artifactId>
		</dependency>
		<dependency>
			<groupId>commons-io</groupId>
			<artifactId>commons-io</artifactId>
			<version>${commons-io.version}</version>
		</dependency>
		<dependency>
			<groupId>com.alibaba</groupId>
			<artifactId>fastjson</artifactId>
			<version>${fastjson.version}</version>
		</dependency>    	
        <!--对json格式的支持 -->
        <dependency>
            <groupId>org.codehaus.jackson</groupId>
            <artifactId>jackson-mapper-asl</artifactId>
            <version>${org.codehaus.jackson.version}</version>
        </dependency>
        <dependency>
            <groupId>com.fasterxml.jackson.core</groupId>
            <artifactId>jackson-databind</artifactId>
        </dependency>   
        <dependency>
            <groupId>com.fasterxml.uuid</groupId>
            <artifactId>java-uuid-generator</artifactId>
            <version>${fasterxml.uuid.version}</version>
        </dependency>  
  	</dependencies>

二、rabbit-api 实现接口定义

1. rabbit-api 所要做的事

首先应该把该项目的整体功能进行抽象,形成一系列接口。这些接口都定义在 rabbit-api 中。

该项目的接口可归类为发送迅速、延迟、可靠三种类型的消息。需要对这三类消息做一个封装。

2. 自定义 Message(放于 rabbit-api 模块下)

既然我们是对消息进行封装,很显然,我们要自己做一个message的实体类,主要把我们定义的消息属性做一个规划,其中一些比较关键的属性如下:

  • messageId 唯一消息id
  • topic 代表exchange主机名
  • routingKey 路由键
  • Map<String,Object> attribates 消息附加属性
  • delayMills  消息延迟时间
  • messageType  消息类型
public class Message implements Serializable {

	private static final long serialVersionUID = 841277940410721237L;

	/* 	消息的唯一ID	*/
	private String messageId;
	
	/*	消息的主题		*/
	private String topic;
	
	/*	消息的路由规则	*/
	private String routingKey = "";
	
	/*	消息的附加属性	*/
	private Map<String, Object> attributes = new HashMap<String, Object>();
	
	/*	延迟消息的参数配置	*/
	private int delayMills;
	
	/*	消息类型:默认为confirm消息类型	*/
	private String messageType = MessageType.CONFIRM;

	public Message() {
	}
	
	public Message(String messageId, String topic, String routingKey, Map<String, Object> attributes, int delayMills) {
		this.messageId = messageId;
		this.topic = topic;
		this.routingKey = routingKey;
		this.attributes = attributes;
		this.delayMills = delayMills;
	}
	
	public Message(String messageId, String topic, String routingKey, Map<String, Object> attributes, int delayMills,
			String messageType) {
		this.messageId = messageId;
		this.topic = topic;
		this.routingKey = routingKey;
		this.attributes = attributes;
		this.delayMills = delayMills;
		this.messageType = messageType;
	}
	
}

其中 messageType 应该包括的类型为:

  •     迅速消息 :不需要保障消息的可靠性,也不需要做confirm确认
  •     确认消息 :不需要保障消息的可靠性,但是做消息confirm确认
  •     可靠消息 :保障消息100%可靠投递,不允许有任何消息丢失。(保障数据所发的消息是原子性的)
public final class MessageType {

	/**
	 * 	迅速消息:不需要保障消息的可靠性, 也不需要做confirm确认
	 */
	public static final String RAPID = "0";
	
	/**
	 * 	确认消息:不需要保障消息的可靠性,但是会做消息的confirm确认
	 */
	public static final String CONFIRM = "1";
	
	/**
	 * 	可靠性消息: 一定要保障消息的100%可靠性投递,不允许有任何消息的丢失
	 * 	PS: 保障数据库和所发的消息是原子性的(最终一致的)
	 */
	public static final String RELIANT = "2";
	
}

对于 Message 对象的创建,这里使用建造者模式,对应的类如下:

/**
 * 	$MessageBuilder 建造者模式
 *
 */
public class MessageBuilder {

    /*
     * Message 中的属性全部拷贝一份
     */
	private String messageId;
	private String topic;
	private String routingKey = "";
	private Map<String, Object> attributes = new HashMap<String, Object>();
	private int delayMills;
	private String messageType = MessageType.CONFIRM;
	
	private MessageBuilder() {
	}

	public static MessageBuilder create() {
		return new MessageBuilder();
	}

	public MessageBuilder withMessageId(String messageId) {
		this.messageId = messageId;
		return this;
	}
	
	public MessageBuilder withTopic(String topic) {
		this.topic = topic;
		return this;
	}
	
	public MessageBuilder withRoutingKey(String routingKey) {
		this.routingKey = routingKey;
		return this;
	}

	public MessageBuilder withAttributes(Map<String, Object> attributes) {
		this.attributes = attributes;
		return this;
	}
	
	public MessageBuilder withAttribute(String key, Object value) {
		this.attributes.put(key, value);
		return this;
	}
	
	public MessageBuilder withDelayMills(int delayMills) {
		this.delayMills = delayMills;
		return this;
	}
	
	public MessageBuilder withMessageType(String messageType) {
		this.messageType = messageType;
		return this;
	}

	public Message build() {
		
		// 1. check messageId 
		if(messageId == null) {
			messageId = UUID.randomUUID().toString();
		}
		// 2. topic is null ,这时应该终止运行
		if(topic == null) {
			throw new MessageRunTimeException("this topic is null");
		}
		Message message = new Message(messageId, topic, routingKey, attributes, delayMills, messageType);
		return message;
	}
	
}

3. 在Spring提供的一场的基础上扩展自己的异常(放于 rabbit-api 模块下)

(1)扩展一般异常:MessageException.java

public class MessageException extends Exception {

	private static final long serialVersionUID = 6347951066190728758L;

	public MessageException() {
		super();
	}
	
	public MessageException(String message) {
		super(message);
	}
	
	public MessageException(String message, Throwable cause) {
		super(message, cause);
	}
	
	public MessageException(Throwable cause) {
		super(cause);
	}
	
}

(2)扩展运行时异常类:MessageRunTimeException.java

public class MessageRunTimeException extends RuntimeException {

	private static final long serialVersionUID = 8651828913888663267L;

	public MessageRunTimeException() {
		super();
	}
	
	public MessageRunTimeException(String message) {
		super(message);
	}
	
	public MessageRunTimeException(String message, Throwable cause) {
		super(message, cause);
	}
	
	public MessageRunTimeException(Throwable cause) {
		super(cause);
	}
}

4. 生产者发送消息的接口api抽象出来(放于 rabbit-api 模块下):

public interface MessageProducer {

	/**
	 * 	$send消息的发送 附带SendCallback回调执行响应的业务逻辑处理
	 * @param message
	 * @param sendCallback
	 * @throws MessageRunTimeException
	 */
	void send(Message message, SendCallback sendCallback) throws MessageRunTimeException;
	
	/**
	 * 	
	 * @param message消息的发送
	 * @throws MessageRunTimeException
	 */
	void send(Message message) throws MessageRunTimeException;
	
	/**
	 * 	$send 消息的批量发送
	 * @param messages
	 * @throws MessageRunTimeException
	 */
	void send(List<Message> messages) throws MessageRunTimeException;
	
}

其中的回调函数 SendCallback.java 定义如下:

/**
 * 	$SendCallback 回调函数处理
 *
 */
public interface SendCallback {

	void onSuccess();
	
	void onFailure();
	
}

5. 消费者监听的接口抽象出来(放于 rabbit-api 模块下)

接口如下:(不过这个接口将来也不打算实现,只会实现生产者发送消息的接口)

/**
 * 	$MessageListener 消费者监听消息
 *
 */
public interface MessageListener {

	void onMessage(Message message);
	
}

三、对 rabbit-api 所定义的接口进行实现

1. 整体架构实现

整体架构的实现需要用到其他几个子模块:rabbit-core-producer、rabbit-common、rabbit-task

(1)添加依赖项

首先是在 rabbit-common 中添加依赖项:

        <!-- 这个rabbit-api就是上面编写的module:rabbit-api-->
        <dependency>
    		<groupId>com.didiok.base.rabbit</groupId>
    		<artifactId>rabbit-api</artifactId>
    		<version>0.0.1-SNAPSHOT</version>  			
  		</dependency>
        <!-- mq的依赖 -->
        <dependency>
  			<groupId>org.springframework.boot</groupId>
  			<artifactId>spring-boot-starter-amqp</artifactId>
  		</dependency>

然后再将 rabbit-common 项目作为依赖项添加到 rabbit-core-producer中的 pom.xml里:

        <!-- rabbit-common作为依赖项添加进 rabbit-core-producer中-->
        <dependency>
    		<groupId>com.didiok.base.rabbit</groupId>
    		<artifactId>rabbit-common</artifactId> 
    		<version>0.0.1-SNAPSHOT</version>			
  		</dependency>

(2)实现自动装配(放于 rabbit-core-producer 模块下)

首先定义一个类,用于 spring 的自动装配。
RabbitProducerAutoConfiguration.java

/**
 * 	$RabbitProducerAutoConfiguration 自动装配 
 *
 */
@Configuration
public class RabbitProducerAutoConfiguration {
​​​​​​​
}

配置一下,将该类加入到自动装配的扫描中,需要在 src/main/resources 下创建 META-INF 文件夹,并在该文件夹中创建 spring.factories文件:

# Auto Configure
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
com.bfxy.rabbit.producer.autoconfigure.RabbitProducerAutoConfiguration

这样如果将该项目作为maven依赖引入到某个Spring 应用,在 该应用启动的时候就会自动加载该类。 

(3)实现 rabbit-api 中的接口类 MessageProducer.java(发送消息的接口,放于 rabbit-core-producer 模块下)

ProducerClient.java

/**
 * 	$ProducerClient 发送消息的实际实现类
 *
 */
@Component
public class ProducerClient implements MessageProducer {

	@Override
	public void send(Message message) throws MessageRunTimeException {
	
	}

	@Override
	public void send(List<Message> messages) throws MessageRunTimeException {
		// TODO Auto-generated method stub
	}
	
	@Override
	public void send(Message message, SendCallback sendCallback) throws MessageRunTimeException {
		// TODO Auto-generated method stub
		
	}
}

ProducerClient.java在包 com.didiok.rabbit.producer 下面,所以需要在 RabbitProducerAutoConfiguration.java 中添加包的扫描 @ComponentScan ,添加后代码如下:

/**
 * 	$RabbitProducerAutoConfiguration 自动装配 
 *
 */
@Configuration
@ComponentScan({"com.didiok.rabbit.producer.*"})
public class RabbitProducerAutoConfiguration {
}

实现 ProducerClient.java 中的第一个方法:

@Component
public class ProducerClient implements MessageProducer {

	@Autowired
	private RabbitBroker rabbitBroker;
	
	@Override
	public void send(Message message) throws MessageRunTimeException {
		Preconditions.checkNotNull(message.getTopic());
		String messageType = message.getMessageType();
		switch (messageType) {
			case MessageType.RAPID:
				rabbitBroker.rapidSend(message);
				break;
			case MessageType.CONFIRM:
				rabbitBroker.confirmSend(message);
				break;
			case MessageType.RELIANT:
				rabbitBroker.reliantSend(message);
				break;
		default:
			break;
		}
	}

	@Override
	public void send(List<Message> messages) throws MessageRunTimeException {

	}
	
	@Override
	public void send(Message message, SendCallback sendCallback) throws MessageRunTimeException {
		// TODO Auto-generated method stub
		
	}
}

发送消息的具体逻辑抽象出来封装在 RabbitBroker.java 中:

/**
 * 	$RabbitBroker 具体发送不同种类型消息的接口
 *
 */
public interface RabbitBroker {
	
	void rapidSend(Message message);
	
	void confirmSend(Message message);
	
	void reliantSend(Message message);
	
	void sendMessages();
	
}

四、三种类型消息发送的具体逻辑实现

1. 迅速消息类型的 的逻辑实现(放于 rabbit-core-producer 模块下)

在RabbitBroker.java的实现类 RabbitBrokerImpl.java中,实现 迅速消息类型 的逻辑:

/**
 * 	$RabbitBrokerImpl 真正的发送不同类型的消息实现类
 *
 */
@Slf4j
@Component
public class RabbitBrokerImpl implements RabbitBroker {

	@Autowired
	private RabbitTemplateContainer rabbitTemplateContainer;

	/**
	 * 	$rapidSend迅速发消息
	 */
	@Override
	public void rapidSend(Message message) {
		message.setMessageType(MessageType.RAPID);
		sendKernel(message);
	}
	
	/**
	 * 	$sendKernel 发送消息的核心方法 使用异步线程池进行发送消息
	 * @param message
	 */
	private void sendKernel(Message message) {
		AsyncBaseQueue.submit((Runnable) () -> {
			CorrelationData correlationData = 
					new CorrelationData(String.format("%s#%s#%s",
							message.getMessageId(),
							System.currentTimeMillis(),
							message.getMessageType()));
			String topic = message.getTopic();
			String routingKey = message.getRoutingKey();
			RabbitTemplate rabbitTemplate = rabbitTemplateContainer.getTemplate(message);
			rabbitTemplate.convertAndSend(topic, routingKey, message, correlationData);
			log.info("#RabbitBrokerImpl.sendKernel# send to rabbitmq, messageId: {}", message.getMessageId());			
		});
	}

	@Override
	public void confirmSend(Message message) {
		
	}

	@Override
	public void reliantSend(Message message) {
		
	}
	
	@Override
	public void sendMessages() {
		
	}
}

这里又使用到了两个类:异步线程队列 AsyncBaseQueue.java 和 RabbitTemplate的池化处理RabbitTemplateContainer.java 。

为了提升发送消息的吞吐量,使用异步线程池发送消息(虽然 rabbitTemplate.convertAndSend()本身就是异步方法,但是这里的异步化是为了让整个匿名表达式都放到异步处理中去,由于这里的匿名表达式代码简单,所以性能提升不大,这里的异步线程池只是作为一个规范建议)。

AsyncBaseQueue.java :

@Slf4j
public class AsyncBaseQueue {

	private static final int THREAD_SIZE = Runtime.getRuntime().availableProcessors();
	
	private static final int QUEUE_SIZE = 10000;
	
	private static ExecutorService senderAsync =
			new ThreadPoolExecutor(THREAD_SIZE,
					THREAD_SIZE,
					60L,
					TimeUnit.SECONDS,
					new ArrayBlockingQueue<Runnable>(QUEUE_SIZE),
					// 创建线程
					new ThreadFactory() {
						@Override
						public Thread newThread(Runnable r) {
							Thread t = new Thread(r);
							// 线程名
							t.setName("rabbitmq_client_async_sender");
							return t;
						}
					},
					// 被拒绝时的处理逻辑
					new java.util.concurrent.RejectedExecutionHandler() {
						@Override
						public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) {
							log.error("async sender is error rejected, runnable: {}, executor: {}", r, executor);
						}
					});
		// 提供对外接口
		public static void submit(Runnable runnable) {
			senderAsync.submit(runnable);
		}	
			
}

通过 @Autowired注入方式实例化的RabbitTemplate是单例,为了提高效率,可以池化处理RabbitTemplate:

  • 一个topic对应一个rabbitTemplate
  • 第一次是创建一个template,等后来topic对应的template存在时就只用从池子中获取;
  • 这也相当于多生产者,并行发送消息,一个topic对应一个生产者,多个topic就对应多个生产者,相比于单例中的单个生产者的效率有所提升。

RabbitTemplateContainer.java

/**
 * 	$RabbitTemplateContainer池化封装
 * 	每一个topic 对应一个RabbitTemplate
 *	1.	提高发送的效率
 * 	2. 	可以根据不同的需求制定化不同的RabbitTemplate, 比如每一个topic 都有自己的routingKey规则
 */
@Slf4j
@Component
public class RabbitTemplateContainer implements RabbitTemplate.ConfirmCallback {

	private Map<String /* TOPIC */, RabbitTemplate> rabbitMap = Maps.newConcurrentMap();
	
	private Splitter splitter = Splitter.on("#");
	
	private SerializerFactory serializerFactory = JacksonSerializerFactory.INSTANCE;
	
	@Autowired
	private ConnectionFactory connectionFactory;
	
	@Autowired
	private MessageStoreService messageStoreService;
	
	public RabbitTemplate getTemplate(Message message) throws MessageRunTimeException {
		Preconditions.checkNotNull(message);
		String topic = message.getTopic();
		RabbitTemplate rabbitTemplate = rabbitMap.get(topic);
        // 池中存在就立即返回,否则就新建一个 newTemplate 
		if(rabbitTemplate != null) {
			return rabbitTemplate;
		}
		log.info("#RabbitTemplateContainer.getTemplate# topic: {} is not exists, create one", topic);
		
		RabbitTemplate newTemplate = new RabbitTemplate(connectionFactory);
		newTemplate.setExchange(topic);
		newTemplate.setRoutingKey(message.getRoutingKey());
		newTemplate.setRetryTemplate(new RetryTemplate());
		
		//	添加序列化反序列化和converter对象
		Serializer serializer = serializerFactory.create();
		GenericMessageConverter gmc = new GenericMessageConverter(serializer);
		RabbitMessageConverter rmc = new RabbitMessageConverter(gmc);
		newTemplate.setMessageConverter(rmc);
		
		String messageType = message.getMessageType();
        // 不是迅速发消息类型的时候,都需要添加消息确认方法:confirm()
		if(!MessageType.RAPID.equals(messageType)) {
			newTemplate.setConfirmCallback(this);
		}
		
		rabbitMap.putIfAbsent(topic, newTemplate);
		
		return rabbitMap.get(topic);
	}

	/**
	 * 	无论是 confirm 消息 还是 reliant 消息 ,发送消息以后 broker都会去回调confirm
	 */
	@Override
	public void confirm(CorrelationData correlationData, boolean ack, String cause) {
		// 	具体的消息应答
		List<String> strings = splitter.splitToList(correlationData.getId());
		String messageId = strings.get(0);
		long sendTime = Long.parseLong(strings.get(1));
		String messageType = strings.get(2);
		if(ack) {
			//	当Broker 返回ACK成功时
			log.info("send message is OK, confirm messageId: {}, sendTime: {}", messageId, sendTime);
		} else {
			log.error("send message is Fail, confirm messageId: {}, sendTime: {}", messageId, sendTime);
			
		}
	}
}

 对于上面代码中的 

        //    添加序列化反序列化和converter对象
        Serializer serializer = serializerFactory.create();
        GenericMessageConverter gmc = new GenericMessageConverter(serializer);
        RabbitMessageConverter rmc = new RabbitMessageConverter(gmc);
        newTemplate.setMessageConverter(rmc);

这块代码是一个设置对象转换方式的操作,这里的转换是指我们自己写的Message实体类和org.springframework.amqp.core.Message之间的转换。并且转换过程可以通过自定义的序列化类进行实现(自定义序列化的意义在于实现指定对象之间的相互转化)。

具体实现可参考:自定义某个Object对象和Spring中的某个Object对象通过序列化和反序列化的方式进行转换(java代码实现)

序列化方法全都定义在 rabbit-common 模块下,其中有:SerializerFactory.java、Serializer.java、 JacksonSerializer.java、JacksonSerializerFactory.java、GenericMessageConverter.java、RabbitMessageConverter.java六个类。 

此外,由于继承了RabbitTemplate.ConfirmCallback类,所以可以重写回调函数,即confirm(CorrelationData correlationData, boolean ack, String cause)方法,在里面添加自己的处理逻辑。

2.  确认类型消息的逻辑实现(放于 rabbit-core-producer 模块下)

基于上面已完成的代码,可以很容易实现 确认类型消息 的发送,直接可以引用前面实现的 sendKernel(message) 方法进行发送,并且 回调函数 confirm()已经在前面实现了,这里就不再说明了。

/**
 * 	$RabbitBrokerImpl 真正的发送不同类型的消息实现类
 *
 */
@Slf4j
@Component
public class RabbitBrokerImpl implements RabbitBroker {

	@Autowired
	private RabbitTemplateContainer rabbitTemplateContainer;
	
	@Autowired
	private MessageStoreService messageStoreService;
	
	@Override
	public void rapidSend(Message message) {
		// 省略......
	}

	/**
	 * 	确认类型的消息发送
	 */
	@Override
	public void confirmSend(Message message) {
		message.setMessageType(MessageType.CONFIRM);
		sendKernel(message);
	}

	/**
	 * 	$sendKernel 发送消息的核心方法 使用异步线程池进行发送消息
	 * @param message
	 */
	private void sendKernel(Message message) {
		AsyncBaseQueue.submit((Runnable) () -> {
			CorrelationData correlationData =
					// 回调函数confirm中需要用到message.getMessageId(), message.getMessageType()。所以可以放在CorrelationData中
					new CorrelationData(String.format("%s#%s#%s",
							message.getMessageId(),
							System.currentTimeMillis(),
							message.getMessageType()));
			String topic = message.getTopic();
			String routingKey = message.getRoutingKey();
			RabbitTemplate rabbitTemplate = rabbitTemplateContainer.getTemplate(message);
			rabbitTemplate.convertAndSend(topic, routingKey, message, correlationData);
			log.info("#RabbitBrokerImpl.sendKernel# send to rabbitmq, messageId: {}", message.getMessageId());			
		});
	}

	@Override
	public void reliantSend(Message message) {
		
	}
	
	@Override
	public void sendMessages() {
		
	}

}

3. 对可靠性发送消息的实现(放于 rabbit-core-producer 模块下)

可靠性发送消息的实现可参考教程 RabbitMQ可靠性消息发送(java实现)

 五、 测试

1. 新建一个工程项目 rabbit-test,并添加我们封装好的基础组件rabbit-core-producer:

	  	<dependency>
	  		<groupId>com.bfxy.base.rabbit</groupId>
	  		<artifactId>rabbit-core-producer</artifactId>
	  		<version>0.0.1-SNAPSHOT</version>
	  	</dependency>

2. 配置文件 application.properties

server.context-path=/test
server.port=8001

spring.application.name=test

# MQ的地址
spring.rabbitmq.addresses=192.168.11.71:5672, 192.168.11.72:5672, 192.168.11.73:5672
spring.rabbitmq.username=guest
spring.rabbitmq.password=guest
spring.rabbitmq.virtual-host=/
spring.rabbitmq.connection-timeout=15000
spring.rabbitmq.publisher-confirms=true
spring.rabbitmq.publisher-returns=true
spring.rabbitmq.template.mandatory=true
spring.rabbitmq.listener.simple.auto-startup=false

## 封装了 ElasticJob 的基础组件 rabbit-task 需要下面这两个 ZooKeeper 配置
elastic.job.zk.serverLists=192.168.11.111:2181,192.168.11.112:2181,192.168.11.113:2181
elastic.job.zk.namespace=elastic-job


3. RabbitMQ和ZooKeeper启动

将RabbitMQ集群和ZooKeeper集群都启动,并在RabbitMQ服务中心(即 ​​​​​​​121.43.153.20:15672界面)手动创建交换机 exchange-1 和队列 queue-1。并通过routingKey:springboot.* 进行绑定。

4. 编写测试类:

ApplicationTests.java

@RunWith(SpringRunner.class)
@SpringBootTest
public class ApplicationTests {

	@Autowired
	private ProducerClient producerClient;
	
	@Test
	public void testProducerClient() throws Exception {
		
		for(int i = 0 ; i < 1; i ++) {
			String uniqueId = UUID.randomUUID().toString();
			Map<String, Object> attributes = new HashMap<>();
			attributes.put("name", "张三");
			attributes.put("age", "18");
			Message message = new Message(
					uniqueId, 
					"exchange-1", 
					"springboot.abc", 
					attributes, 
					0);
			message.setMessageType(MessageType.RELIANT);
//			message.setDelayMills(15000);
			producerClient.send(message);			
		}

		Thread.sleep(100000);
	}
	
	
}

然后也可以测试一下发送失败的场景:使用一个不存在的交换机进行发送,就会发送失败,比如将上面的交换机名称修改成 exchange-2,然后重新执行发送即可。并查看 数据库中表 broker_message 的状态 status 的值,是成功还是失败。

(补充)六、 批量消息发送和延迟消息发送的实现

1. 批量消息发送

(1)工具类编写

这里对于批量发送的消息类型默认为 迅速消息类型(MessageType.RAPID).

为了在使用send(List<Message> messages)批量发消息之前,可以在该链路请求过程中任何时候添加要发送的消息,然后打包成一个请求执行批量发送。因而使用线程变量 ThreadLocal 来暂存Message,等到执行send(List<Message> messages)时,再从ThreadLocal中取出Message,从而实现一次性批量发送。

ThreadLocal是一个类似于hashmap的二级map,与当前线程是一个弱引用关系,这里借助ThreadLocal实现消息的缓存,对其进行简单封装一下:

MessageHolder.java

public class MessageHolder {

	private List<Message> messages = Lists.newArrayList();
	
	@SuppressWarnings({"rawtypes", "unchecked"})
	public static final ThreadLocal<MessageHolder> holder = new ThreadLocal() {
		@Override
		protected Object initialValue() {
			return new MessageHolder();
		}
	};

	/**
	 *
	 * @param message
	 */
	public static void add(Message message) {
		holder.get().messages.add(message);
	}

	/**
	 * 从 ThreadLocal 取出数据,并清空 ThreadLocal
	 * @return
	 */
	public static List<Message> clear() {
		List<Message> tmp = Lists.newArrayList(holder.get().messages);
		holder.remove();
		return tmp;
	}
	
}

这里对于发送消息也是使用异步线程去发送,为了和之前单个消息发送的异步线程队列区分开,这里再创建一个异步线程队列,但是其中的逻辑和之前的线程队列 AsyncBaseQueue 一样的,只是换了一个队列名称而已。新的线程队列:

MessageHolderAyncQueue.java

@Slf4j
public class MessageHolderAyncQueue {

	private static final int THREAD_SIZE = Runtime.getRuntime().availableProcessors();
	
	private static final int QUEUE_SIZE = 10000;
	
	private static ExecutorService senderAsync =
			new ThreadPoolExecutor(THREAD_SIZE,
					THREAD_SIZE,
					60L,
					TimeUnit.SECONDS,
					new ArrayBlockingQueue<Runnable>(QUEUE_SIZE),
					new ThreadFactory() {
						@Override
						public Thread newThread(Runnable r) {
							Thread t = new Thread(r);
							t.setName("rabbitmq_client_async_sender");
							return t;
						}
					},
					new java.util.concurrent.RejectedExecutionHandler() {
						@Override
						public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) {
							log.error("async sender is error rejected, runnable: {}, executor: {}", r, executor);
						}
					});
			
		public static void submit(Runnable runnable) {
			senderAsync.submit(runnable);
		}	
}

(2)批量发送实现

现在开始编写发送的逻辑:首先是改造最外层的发送方法,即在 ProducerClient.java 中加入批量发送代码:

@Component
public class ProducerClient implements MessageProducer {

	@Autowired
	private RabbitBroker rabbitBroker;
	
	@Override
	public void send(Message message) throws MessageRunTimeException {
		// 省略...
	}

	/**
	 * 	批量发送消息
	 */
	@Override
	public void send(List<Message> messages) throws MessageRunTimeException {
		messages.forEach( message -> {
			message.setMessageType(MessageType.RAPID);
			// 往 ThreadLocal 中暂存消息
			MessageHolder.add(message);
		});
		rabbitBroker.sendMessages();
	}
	
	@Override
	public void send(Message message, SendCallback sendCallback) throws MessageRunTimeException {
		// TODO Auto-generated method stub
		
	}

}

然后是实现 rabbitBroker.sendMessages() 方法:

RabbitBrokerImpl.java

/**
 * 	$RabbitBrokerImpl 真正的发送不同类型的消息实现类
 * @author Alienware
 *
 */
@Slf4j
@Component
public class RabbitBrokerImpl implements RabbitBroker {

	@Autowired
	private RabbitTemplateContainer rabbitTemplateContainer;
	
	@Autowired
	private MessageStoreService messageStoreService;
	
	@Override
	public void reliantSend(Message message) {
		// 省略...
	}
	
	/**
	 * 	$rapidSend迅速发消息
	 */
	@Override
	public void rapidSend(Message message) {
		// 省略...
	}
	
	/**
	 * 	$sendKernel 发送消息的核心方法 使用异步线程池进行发送消息
	 * @param message
	 */
	private void sendKernel(Message message) {
		// 省略...
	}

	@Override
	public void confirmSend(Message message) {
		// 省略...
	}

	@Override
	public void sendMessages() {
		// 从 ThreadLocal 中取出全部消息
		List<Message> messages = MessageHolder.clear();
		// 异步线程循环发送
		messages.forEach(message -> {
			MessageHolderAyncQueue.submit((Runnable) () -> {
				CorrelationData correlationData = 
						new CorrelationData(String.format("%s#%s#%s",
								message.getMessageId(),
								System.currentTimeMillis(),
								message.getMessageType()));
				String topic = message.getTopic();
				String routingKey = message.getRoutingKey();
				RabbitTemplate rabbitTemplate = rabbitTemplateContainer.getTemplate(message);
				rabbitTemplate.convertAndSend(topic, routingKey, message, correlationData);
				log.info("#RabbitBrokerImpl.sendMessages# send to rabbitmq, messageId: {}", message.getMessageId());			
			});			
		});
	}

}

2. 延迟消息发送

首先需要安装一个延迟发送的插件,教程可参考:RabbitMQ集群环境搭建-镜像模式 中的第三步骤。

代码实现:

(1)发送消息时指定延迟时长,这里设置延迟15秒钟:

@RunWith(SpringRunner.class)
@SpringBootTest
public class ApplicationTests {

	@Autowired
	private ProducerClient producerClient;
	
	@Test
	public void testProducerClient2() throws Exception {
		
		for(int i = 0 ; i < 1; i ++) {
			String uniqueId = UUID.randomUUID().toString();
			Map<String, Object> attributes = new HashMap<>();
			attributes.put("name", "张三");
			attributes.put("age", "18");
			Message message = new Message(
					uniqueId, 
					"delay-exchange", 
					"delay.abc", 
					attributes, 
					15000); // 延迟15秒钟
			message.setMessageType(MessageType.RELIANT);
			producerClient.send(message);			
		}

		Thread.sleep(100000);
	}
}

(2)添加延迟属性

在我们自定义的Message对象和Spring中的org.springframework.amqp.core.Message对象转换的过程中,将这个延时时长添加到附加属性 Properties 中,如下代码:

自定义的Message对象和Spring中的org.springframework.amqp.core.Message对象转换的逻辑可参考教程:自定义某个Object对象和Spring中的某个Object对象通过序列化和反序列化的方式进行转换

RabbitMessageConverter.java

public class RabbitMessageConverter implements MessageConverter {

	private GenericMessageConverter delegate;
	
	public RabbitMessageConverter(GenericMessageConverter genericMessageConverter) {
		// 省略...
	}
	
	@Override
	public Message toMessage(Object object, MessageProperties messageProperties) throws MessageConversionException {
		com.bfxy.rabbit.api.Message message = (com.bfxy.rabbit.api.Message)object;
        // 延迟消息:把延迟时长作为 x-delay 的属性值,添加到 messageProperties 中
		messageProperties.setDelay(message.getDelayMills());
		return this.delegate.toMessage(object, messageProperties);
	}

	@Override
	public Object fromMessage(Message message) throws MessageConversionException {
		// 省略...
	}

}

把延迟时长作为 x-delay 的属性值,添加到 messageProperties 中,这样就实现了延迟消息的发送。

;