Skip to main content

moregeek program

使用 redis 进行消息传递_spring认证的博客-多极客编程

本指南将引导您完成使用 Spring Data Redis 发布和订阅通过 Redis 发送的消息的过程。

使用 Redis 进行消息传递_redis

你将建造什么

您将构建一个应用程序,该应用程序用于​​StringRedisTemplate​​​发布字符串消息并使用 POJO 订阅该消息​​MessageListenerAdapter​​。

使用 Spring Data Redis 作为发布消息的方式可能听起来很奇怪,但是,正如您将发现的那样,Redis 不仅提供了 NoSQL 数据存储,还提供了消息传递系统。

你需要什么

  • 约15分钟
  • 最喜欢的文本编辑器或 IDE
  • JDK 1.8或更高版本
  • Gradle 4+或Maven 3.2+
  • 您还可以将代码直接导入 IDE:
  • 弹簧工具套件 (STS)
  • IntelliJ IDEA
  • Redis 服务器

如何完成本指南

像大多数 Spring入门指南一样,您可以从头开始并完成每个步骤,也可以绕过您已经熟悉的基本设置步骤。无论哪种方式,您最终都会得到工作代码。

从头开始,请继续建立 Redis 服务器。

跳过基础知识,请执行以下操作:

  • 下载并解压本指南的源代码库,或使用Git克隆它:git clone https://github.com/spring-guides/gs-messaging-redis.git
  • 光盘进入gs-messaging-redis/initial
  • 跳转到从 Spring Initializr 开始。

完成后,您可以对照中的代码检查结果​​gs-messaging-redis/complete​​。

建立一个 Redis 服务器

在构建消息传递应用程序之前,您需要设置将处理接收和发送消息的服务器。

Redis 是一个开源的、BSD 许可的键值对数据存储,它还附带一个消息传递系统。该服务器可在https://redis.io/download免费获得。您可以手动下载它,或者,如果您使用 Mac 和 Homebrew,则可以在终端窗口中运行以下命令:

brew install redis复制

解压 Redis 后,您可以通过运行以下命令以默认设置启动它:

redis-server复制

您应该会看到类似于以下内容的消息:

[35142] 01 May 14:36:28.939 # Warning: no config file specified, using the default config. In order to specify a config file use redis-server /path/to/redis.conf
[35142] 01 May 14:36:28.940 * Max number of open files set to 10032
_._
_.-``__ ''-._
_.-`` `. `_. ''-._ Redis 2.6.12 (00000000/0) 64 bit
.-`` .-```. ```\/ _.,_ ''-._
( ' , .-` | `, ) Running in stand alone mode
|`-._`-...-` __...-.``-._|'` _.-'| Port: 6379
| `-._ `._ / _.-' | PID: 35142
`-._ `-._ `-./ _.-' _.-'
|`-._`-._ `-.__.-' _.-'_.-'|
| `-._`-._ _.-'_.-' | https://redis.io
`-._ `-._`-.__.-'_.-' _.-'
|`-._`-._ `-.__.-' _.-'_.-'|
| `-._`-._ _.-'_.-' |
`-._ `-._`-.__.-'_.-' _.-'
`-._ `-.__.-' _.-'
`-._ _.-'
`-.__.-'

[35142] 01 May 14:36:28.941 # Server started, Redis version 2.6.12
[35142] 01 May 14:36:28.941 * The server is now ready to accept connections on port 6379复制

从 Spring Initializr 开始

您可以使用这个预先初始化的项目并单击 Generate 下载 ZIP 文件。此项目配置为适合本教程中的示例。

手动初始化项目:

  1. 导航到https://start.spring.io。该服务提取应用程序所需的所有依赖项,并为您完成大部分设置。
  2. 选择 Gradle 或 Maven 以及您要使用的语言。本指南假定您选择了 Java。
  3. 单击Dependencies并选择Spring Data Redis
  4. 单击生成
  5. 下载生成的 ZIP 文件,该文件是根据您的选择配置的 Web 应用程序的存档。
如果您的 IDE 具有 Spring Initializr 集成,您可以从您的 IDE 完成此过程。

你也可以从 Github 上 fork 项目并在你的 IDE 或其他编辑器中打开它。

创建 Redis 消息接收器

在任何基于消息传递的应用程序中,都有消息发布者和消息接收者。要创建消息接收器,请使用响应消息的方法实现接收器,如以下示例 (from ​​src/main/java/com/example/messagingredis/Receiver.java​​) 所示:

package com.example.messagingredis;

import java.util.concurrent.atomic.AtomicInteger;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class Receiver {
private static final Logger LOGGER = LoggerFactory.getLogger(Receiver.class);

private AtomicInteger counter = new AtomicInteger();

public void receiveMessage(String message) {
LOGGER.info("Received <" + message + ">");
counter.incrementAndGet();
}

public int getCount() {
return counter.get();
}
}复制

这​​Receiver​​​是一个 POJO,它定义了接收消息的方法。当您将 注册​​Receiver​​为消息侦听器时,您可以随意命名消息处理方法。

出于演示目的,接收方正在对收到的消息进行计数。这样,它可以在收到消息时发出信号。

注册监听器并发送消息

Spring Data Redis 提供了使用 Redis 发送和接收消息所需的所有组件。具体来说,需要配置:

  • 连接工厂
  • 消息侦听器容器
  • 一个 Redis 模板

您将使用 Redis 模板发送消息,并将​​Receiver​​向消息侦听器容器注册,以便它接收消息。连接工厂同时驱动模板和消息侦听器容器,让它们连接到 Redis 服务器。

此示例使用 Spring Boot 的 default ​​RedisConnectionFactory​​,它的一个实例​​JedisConnectionFactory​​基于Jedis Redis 库。连接工厂被注入到消息侦听器容器和 Redis 模板中,如以下示例(来自​​src/main/java/com/example/messagingredis/MessagingRedisApplication.java​​)所示:

package com.example.messagingredis;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.data.redis.connection.RedisConnectionFactory;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.data.redis.listener.PatternTopic;
import org.springframework.data.redis.listener.RedisMessageListenerContainer;
import org.springframework.data.redis.listener.adapter.MessageListenerAdapter;

@SpringBootApplication
public class MessagingRedisApplication {

private static final Logger LOGGER = LoggerFactory.getLogger(MessagingRedisApplication.class);

@BeanRedisMessageListenerContainer container(RedisConnectionFactory connectionFactory,MessageListenerAdapter listenerAdapter) {

RedisMessageListenerContainer container = new RedisMessageListenerContainer();
container.setConnectionFactory(connectionFactory);
container.addMessageListener(listenerAdapter, new PatternTopic("chat"));

return container;}

@BeanMessageListenerAdapter listenerAdapter(Receiver receiver) {return new MessageListenerAdapter(receiver, "receiveMessage");}

@BeanReceiver receiver() {return new Receiver();}

@BeanStringRedisTemplate template(RedisConnectionFactory connectionFactory) {return new StringRedisTemplate(connectionFactory);}

public static void main(String[] args) throws InterruptedException {

ApplicationContext ctx = SpringApplication.run(MessagingRedisApplication.class, args);

StringRedisTemplate template = ctx.getBean(StringRedisTemplate.class);Receiver receiver = ctx.getBean(Receiver.class);

while (receiver.getCount() == 0) {

LOGGER.info("Sending message...");template.convertAndSend("chat", "Hello from Redis!");Thread.sleep(500L);}

System.exit(0);}
}复制

方法中定义的bean在定义​​listenerAdapter​​的消息监听器容器中注册为消息监听器​​container​​,将监听该​​chat​​主题的消息。因为​​Receiver​​该类是 POJO,所以需要将其包装在实现​​MessageListener​​接口的消息侦听器适配器中(这是 所需的​​addMessageListener()​​)。消息侦听器适配器还配置为在消息到达时调用该​​receiveMessage()​​方法。​​Receiver​

连接工厂和消息侦听器容器 bean 是您侦听消息所需的全部。要发送消息,您还需要一个 Redis 模板。在这里,它是一个配置为 a 的 bean ​​StringRedisTemplate​​,其实现​​RedisTemplate​​侧重于 Redis 的常见用途,其中键和值都是​​String​​实例。

该​​main()​​方法通过创建 Spring 应用程序上下文来启动一切。然后应用程序上下文启动消息侦听器容器,消息侦听器容器 bean 开始侦听消息。然后该​​main()​​方法从应用程序上下文中检索​​StringRedisTemplate​​bean 并使用它来发送​​Hello from Redis!​​有关​​chat​​主题的消息。最后,它关闭 Spring 应用程序上下文,应用程序结束。

构建一个可执行的 JAR

您可以使用 Gradle 或 Maven 从命令行运行应用程序。您还可以构建一个包含所有必要依赖项、类和资源的单个可执行 JAR 文件并运行它。构建可执行 jar 可以在整个开发生命周期、跨不同环境等中轻松地作为应用程序交付、版本化和部署服务。

如果您使用 Gradle,则可以使用​​./gradlew bootRun​​. 或者,您可以使用构建 JAR 文件​​./gradlew build​​,然后运行 ​​JAR 文件,如下所示:

java -jar build/libs/gs-messaging-redis-0.1.0.jar

如果您使用 Maven,则可以使用​​./mvnw spring-boot:run​​. 或者,您可以使用构建 JAR 文件,​​./mvnw clean package​​然后运行该 JAR 文件,如下所示:

java -jar 目标/gs-messaging-redis-0.1.0.jar

此处描述的步骤创建了一个可运行的 JAR。您还可以构建经典的 WAR 文件。

您应该会看到类似于以下内容的输出:

  .   ____          _            __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v2.1.8.RELEASE)

2019-09-23 12:57:11.578 INFO 35396 --- [ main] c.e.m.MessagingRedisApplication : Starting MessagingRedisApplication on Jays-MBP with PID 35396 (/Users/j/projects/guides/gs-messaging-redis/complete/target/classes started by j in /Users/j/projects/guides/gs-messaging-redis/complete)
2019-09-23 12:57:11.581 INFO 35396 --- [ main] c.e.m.MessagingRedisApplication : No active profile set, falling back to default profiles: default
2019-09-23 12:57:11.885 INFO 35396 --- [ main] .s.d.r.c.RepositoryConfigurationDelegate : Multiple Spring Data modules found, entering strict repository configuration mode!
2019-09-23 12:57:11.887 INFO 35396 --- [ main] .s.d.r.c.RepositoryConfigurationDelegate : Bootstrapping Spring Data repositories in DEFAULT mode.
2019-09-23 12:57:11.914 INFO 35396 --- [ main] .s.d.r.c.RepositoryConfigurationDelegate : Finished Spring Data repository scanning in 13ms. Found 0 repository interfaces.
2019-09-23 12:57:12.685 INFO 35396 --- [ container-1] io.lettuce.core.EpollProvider : Starting without optional epoll library
2019-09-23 12:57:12.685 INFO 35396 --- [ container-1] io.lettuce.core.KqueueProvider : Starting without optional kqueue library
2019-09-23 12:57:12.848 INFO 35396 --- [ main] c.e.m.MessagingRedisApplication : Started MessagingRedisApplication in 1.511 seconds (JVM running for 3.685)
2019-09-23 12:57:12.849 INFO 35396 --- [ main] c.e.m.MessagingRedisApplication : Sending message...
2019-09-23 12:57:12.861 INFO 35396 --- [ container-2] com.example.messagingredis.Receiver : Received <Hello from Redis!>复制

概括

恭喜!您刚刚使用 Spring 和 Redis 开发了一个发布和订阅应用程序。

©著作权归作者所有:来自51CTO博客作者Spring_Learnin的原创作品,请联系作者获取转载授权,否则将追究法律责任

spring集成junit单元测试框架_是温度呀的博客-多极客编程

一.JUnit介绍JUnit是Java中最有名的单元测试框架,用于编写和运行可重复的测试,多数Java的开发环境都已经集成了JUnit作为单元测试的工具。好的单元测试能极大的提高开发效率和代码质量。测试类命名规则:被测试类+Test,如UserServiceTest测试用例命名规则:test+用例方法,如testGetMaven导入junit、sprint-test 、json-path相关测试包

使用 rabbitmq 进行消息传递_spring认证的博客-多极客编程

本指南将引导您完成设置发布和订阅消息的 RabbitMQ AMQP 服务器以及创建 Spring Boot 应用程序以与该 RabbitMQ 服务器交互的过程。你将建造什么您将构建一个应用程序,该应用程序使用 Spring AMQP 发布消息​​RabbitTemplate​​并使用​​MessageListenerAdapter​​.你需要什么约15分钟最喜欢的文本编辑器或 IDEJDK 11或

基于 stringredistemplate 封装 缓存工具类_perceus.的博客-多极客编程

(目录) 封装Redis工具类 ( 巨干 ,注意食用) 基于StringRedisTemplate封装一个缓存工具类,满足下列需求: 方法1:将任意Java对象序列化为json并存储在string类型的key中,并且可以设置TTL过期时间 方法2:将任意Java对象序列化为json并存储在string类型的key中,并且可以设置逻辑过期时间,用于处理缓存击穿问题 方法3:根据指定的

如何使用 junit + mockito 实践单元测试_是温度呀的博客-多极客编程

一、前言相信做过开发的同学,都多多少少写过下面的代码,很长一段时间我一直以为这就是单元测试...@SpringBootTest@RunWith(SpringRunner.class)public class UnitTest1 { @Autowired private UnitService unitService; @Test public void test(){

用好java中的函数式接口,轻松从通用代码框架中剥离掉业务定制逻辑_架构悟道的博客-多极客编程

大家好,又见面了。 今天我们一起聊一聊JAVA中的函数式接口。那我们首先要知道啥是函数式接口、它和JAVA中普通的接口有啥区别?其实函数式接口也是一个Interface类,是一种比较特殊的接口类,这个接口类有且仅有一个抽象方法(但是可以有其余的方法,比如default方法)。 当然,我们看源码的时候,会发现JDK中提供的函数式接口,都会携带一个 @FunctionalFunction注解,这个注释

【优惠卷秒杀系统设计】全局唯一id生成,秒杀下单,分析解决库存超卖(乐观锁)、实现一人一单秒杀业务(对用户id加锁)、集群模式下并发问题_perceus.的博客-多极客编程

(目录) 优惠卷秒杀 使用Redis的计数器功能, 结合Lua完成高性能的redis操作,同时学会Redis分布式锁的原理,包括Redis的三种消息队列 优惠卷秒杀 1. 全局唯一ID 每个店铺都可以发布优惠券: 当用户抢购时,就会生成订单并保存到tb_voucher_order这张表中,而订单表如果使用数据库自增ID就存在一些问题: id的规律性太明显 受单表数据量的限制

[python编程:从入门到实践] 变量&字符串_wx5d1daeb3cba49的博客-多极客编程

在本章中,你将学习可在Python程序中使用的各种数据,还将学习如何将数据存储到变量中,以及如何在程序中使用这些变量。 一、运行过程 Python是解释型语言,Python解释器来解释运行程序,解释运行程序(边解释边执行),非编译运行(执行编译后的程序文件) 二、变量 每个变量都存储了一个值——与变量相关联的信息。 变量名只能包含字母、数字和下划线。变量名可以字母或下划线打头,但不能以数字

go语言中没有结构化并发?_auula的博客-多极客编程

本文原文地址在本博主博客,点击链接前往:Go语言中有没有结构化并发? 什么是结构化并发?日常开发中我们编写的最多就是多线程程序,服务器端应用更是如此,传统的方式都是依靠着操作系统提供的1:1线程方式进行请求处理这对于管理和复用线程有很多挑战,如果一个普通线程大小2MB那么开启1000个线程,几乎是无法完成的,并且管理这些线程的状态也是很复杂的。今天这篇文章要介绍的是结构化并发,就是为解决并发

spring集成junit单元测试框架_是温度呀的博客-多极客编程

一.JUnit介绍JUnit是Java中最有名的单元测试框架,用于编写和运行可重复的测试,多数Java的开发环境都已经集成了JUnit作为单元测试的工具。好的单元测试能极大的提高开发效率和代码质量。测试类命名规则:被测试类+Test,如UserServiceTest测试用例命名规则:test+用例方法,如testGetMaven导入junit、sprint-test 、json-path相关测试包

动态内存开辟与柔性数组详解_萌新的日常的博客-多极客编程

(文章目录) 一、动态内存函数 1.malloc函数 size代表字节数 如果 开辟空间成功 则返回这块空间的地址 如果 开辟空间失败 则返回NULL 正常来说 创建10个整形空间 应为 void*p=void *malloc(10 sizoef(int)); 但是由于void 解引用会报错 所以 (int * )p=(int * )malloc(10sizeof(int)); #incl

如何使用 junit + mockito 实践单元测试_是温度呀的博客-多极客编程

一、前言相信做过开发的同学,都多多少少写过下面的代码,很长一段时间我一直以为这就是单元测试...@SpringBootTest@RunWith(SpringRunner.class)public class UnitTest1 { @Autowired private UnitService unitService; @Test public void test(){

算法基础(二)| 高精度算法详解_wx62e40d60030b6的博客-多极客编程

⭐写在前面的话:本系列文章旨在复习算法刷题中常用的基础算法与数据结构,配以详细的图例解释,总结相应的代码模板,同时结合例题以达到最佳的学习效果。本专栏面向算法零基础但有一定的C++基础的学习者。若C++基础不牢固,可参考:10min快速回顾C++语法,进行语法复习。🔥本文已收录于算法基础系列专栏: 算法基础教程 免费订阅,持续更新。高精度加法适用于c++,java和python没有这个问题,因为j