快速上手Spring Cloud 七:事件驱动架构与Spring Cloud
快速上手Spring Cloud 一:Spring Cloud 简介
快速上手Spring Cloud 二:核心组件解析
快速上手Spring Cloud 三:API网关深入探索与实战应用
快速上手Spring Cloud 四:微服务治理与安全
快速上手Spring Cloud 五:Spring Cloud与持续集成/持续部署(CI/CD)
快速上手Spring Cloud 六:容器化与微服务化
快速上手Spring Cloud 七:事件驱动架构与Spring Cloud
快速上手Spring Cloud 八:微服务架构中的数据管理
快速上手Spring Cloud 九:服务间通信与消息队列
快速上手Spring Cloud 十:Spring Cloud与微前端
快速上手Spring Cloud 十一:微服务架构下的安全与权限管理
快速上手Spring Cloud 十二:与云原生不得不说的故事
文章目录
- 一、事件驱动架构在微服务中的应用
- 1、事件驱动架构的核心思想
- 2、事件驱动架构的优势
- 3、Spring Cloud在事件驱动架构中的应用
- 二、Spring Cloud Stream与事件驱动架构的集成
- 1、Spring Cloud Stream的核心概念
- 2、Spring Cloud Stream与事件驱动的集成
- 3、配置与绑定
- 4、优势与注意事项
- 三、使用Spring Cloud Bus实现消息驱动微服务
- 1. 引入Spring Cloud Bus依赖
- 2. 配置消息代理
- 3. 发送消息
- 4. 监听消息
- 5. 刷新配置
- 总结
在微服务架构中,事件驱动架构是一种重要的设计模式,它允许微服务之间通过发布和订阅事件进行通信。这种通信方式具有松耦合、异步和可扩展的特点,非常适合处理分布式系统中的复杂业务逻辑。Spring Cloud作为一套微服务解决方案,提供了丰富的组件来支持事件驱动架构的实现。本文将深入探讨事件驱动架构在微服务中的应用需求,以及如何通过Spring Cloud Stream和Spring Cloud Bus实现事件驱动微服务的集成。
一、事件驱动架构在微服务中的应用
事件驱动架构在微服务中的应用
随着企业业务的飞速发展和技术的不断进步,微服务架构成为了应对复杂业务系统的得力助手。但随着微服务数量的不断增多和服务间交互的日益复杂,传统的请求/响应通信模式逐渐暴露出种种弊端。为了克服这些挑战,事件驱动架构应运而生,成为微服务通信的主流方式。
1、事件驱动架构的核心思想
事件驱动架构的核心思想是“发布-订阅”。在这种架构中,微服务将需要传递的信息封装成事件进行发布,而其他对此类信息感兴趣的微服务则通过订阅这些事件来接收信息。这种机制有效地降低了微服务之间的耦合度,使得每个服务都可以独立地运行和扩展,从而提高了整个系统的可扩展性和灵活性。
2、事件驱动架构的优势
-
降低耦合度
事件驱动架构使得微服务之间的依赖关系变得更为松散。每个微服务只需要关注自己需要处理的事件,而无需了解其他微服务的实现细节。这种松耦合的特性使得系统在面临变更时更加稳健,减少了因某个服务变动而引发的连锁反应。 -
提高可扩展性
在事件驱动架构中,微服务的扩展变得更为简单和灵活。当某个服务的负载增加时,我们只需要增加处理该事件的服务实例即可,而无需对整个系统进行大规模的调整。这种动态扩展的能力使得系统能够更好地应对业务增长带来的挑战。 -
实现异步通信
事件驱动架构支持异步通信模式,即微服务之间不需要实时等待对方的响应。这种机制使得系统能够处理大量的并发请求,提高了系统的吞吐量和响应速度。同时,异步通信也降低了服务间的耦合度,使得系统更加健壮和可靠。
3、Spring Cloud在事件驱动架构中的应用
Spring Cloud作为一套成熟的微服务解决方案,为事件驱动架构的实现提供了强大的支持。下面我们将结合Spring Cloud的相关组件,深入探讨如何在微服务中实现事件驱动架构。
- 使用Spring Cloud Stream实现事件发布与订阅
Spring Cloud Stream是一个构建消息驱动微服务的框架,它简化了与消息中间件(如RabbitMQ、Kafka等)的集成。通过定义输入通道和输出通道,我们可以轻松地将微服务中的事件发布到消息中间件,并订阅其他微服务发布的事件。
示例代码:
// 定义事件发布者
@EnableBinding(Source.class)
public class EventPublisher {@Autowiredprivate Source source;public void publishEvent(MyEvent event) {source.output().send(MessageBuilder.withPayload(event).build());}
}// 定义事件订阅者
@EnableBinding(Sink.class)
public class EventSubscriber {@StreamListener(Sink.INPUT)public void handleEvent(MyEvent event) {// 处理事件的逻辑}
}
在上面的代码中,我们定义了一个事件发布者EventPublisher
和一个事件订阅者EventSubscriber
。EventPublisher
通过Source
接口的output()
方法将MyEvent
对象作为消息发送到消息中间件,而EventSubscriber
则通过@StreamListener
注解监听消息中间件中的消息,并在接收到消息时调用handleEvent
方法处理事件。
- 使用Spring Cloud Bus实现服务间通信
Spring Cloud Bus是一个轻量级的消息总线,它利用消息中间件作为通信媒介,实现了微服务之间的广播和监听功能。通过Spring Cloud Bus,我们可以方便地实现微服务之间的状态更新、配置刷新等操作。
示例代码:
@Service
public class BusService {@Autowiredprivate Bus bus;public void refreshConfig() {bus.send("/refresh", new GenericMessage<>("refresh"));}
}
在上面的代码中,我们注入了一个Bus
对象,并通过调用其send
方法向指定的目的地(如/refresh
)发送消息。其他订阅了该目的地的微服务将能够接收到该消息,并执行相应的处理逻辑(如刷新配置)。
事件驱动架构为微服务之间的通信提供了一种高效、灵活且可扩展的解决方案。通过Spring Cloud的相关组件,我们可以轻松地构建基于事件驱动的微服务架构,实现微服务之间的松耦合、异步通信和动态扩展。这种架构不仅提高了系统的可扩展性和灵活性,还降低了维护成本和风险,为企业的快速发展提供了有力的支持。
二、Spring Cloud Stream与事件驱动架构的集成
Spring Cloud Stream作为Spring Cloud生态系统中的一个关键组件,为微服务之间的事件驱动通信提供了强大的支持。它简化了消息中间件(如RabbitMQ、Kafka等)的集成,使得开发者能够更容易地构建基于事件驱动的微服务应用。
1、Spring Cloud Stream的核心概念
Spring Cloud Stream的核心概念包括输入通道(Input Channel)、输出通道(Output Channel)以及消息绑定器(Binder)。输入通道用于接收来自消息中间件的消息,而输出通道则用于发送消息到消息中间件。消息绑定器则负责将通道与具体的消息中间件进行绑定。
2、Spring Cloud Stream与事件驱动的集成
在事件驱动架构中,Spring Cloud Stream充当了消息传递的桥梁。通过将业务事件封装成消息,并利用Spring Cloud Stream进行发布和订阅,微服务之间可以实现异步、松耦合的通信。
1. 事件发布
微服务作为事件发布者,通过输出通道将事件消息发送到消息中间件。在Spring Cloud Stream中,开发者可以通过注解或编程方式定义输出通道,并调用相应的发送方法将事件消息发送到消息中间件。
示例代码:
@EnableBinding(Source.class)
public class EventPublisher {@Autowiredprivate Source source;public void publishEvent(MyEvent event) {Message<MyEvent> message = MessageBuilder.withPayload(event).build();source.output().send(message);}
}
在上述示例中,EventPublisher
类通过@EnableBinding(Source.class)
注解启用了消息绑定,并注入了Source
类型的source
对象。publishEvent
方法创建了一个包含事件负载的Message
对象,并通过source.output().send(message)
将其发送到消息中间件。
2. 事件订阅
微服务作为事件订阅者,通过输入通道从消息中间件接收事件消息,并执行相应的处理逻辑。在Spring Cloud Stream中,开发者可以使用@StreamListener
注解来监听输入通道上的消息,并定义处理方法。
示例代码:
@EnableBinding(Sink.class)
public class EventSubscriber {@StreamListener(Sink.INPUT)public void handleEvent(MyEvent event) {// 处理事件的逻辑System.out.println("Received event: " + event);}
}
在上述示例中,EventSubscriber
类通过@EnableBinding(Sink.class)
注解启用了消息绑定。handleEvent
方法使用@StreamListener(Sink.INPUT)
注解来监听输入通道上的消息。当接收到消息时,该方法会被调用,并可以执行相应的处理逻辑。
3、配置与绑定
为了使Spring Cloud Stream正常工作,需要进行一些配置和绑定操作。这包括指定消息中间件的连接信息、定义通道与消息中间件的映射关系等。这些配置可以通过配置文件(如application.yml
或application.properties
)或编程方式进行设置。
此外,Spring Cloud Stream还提供了消息转换器(Message Converter)和消息序列化器(Message Serializer)等组件,用于处理消息的格式转换和序列化操作,以确保消息的正确传输和处理。
4、优势与注意事项
通过集成Spring Cloud Stream与事件驱动架构,我们可以获得以下优势:
- 简化开发:Spring Cloud Stream提供了统一的编程模型,简化了消息中间件的集成过程,降低了开发难度。
- 松耦合与异步通信:事件驱动架构通过异步消息传递实现了微服务之间的松耦合通信,提高了系统的可扩展性和可靠性。
- 灵活性:Spring Cloud Stream支持多种消息中间件,开发者可以根据实际需求选择合适的中间件进行集成。
然而,在集成过程中也需要注意以下事项:
- 消息一致性:确保在分布式环境下消息的可靠传递和一致性处理,避免消息丢失或重复消费等问题。
- 错误处理与重试机制:合理设计错误处理逻辑和重试机制,以应对网络故障、中间件异常等情况。
- 性能调优:根据系统需求对消息中间件进行性能调优,确保消息传递的高效性和实时性。
Spring Cloud Stream与事件驱动架构的集成为企业构建分布式、可扩展的微服务应用提供了有力的支持。通过合理的配置和设计,我们可以充分发挥事件驱动架构的优势,实现高效、可靠的微服务通信。
三、使用Spring Cloud Bus实现消息驱动微服务
Spring Cloud Bus是一个轻量级的消息总线,它整合了Spring Cloud Stream的功能,为微服务架构中的服务间通信提供了便利。Spring Cloud Bus能够利用消息代理(如RabbitMQ、Kafka等)来在微服务之间广播状态变化和其他事件,使得服务能够响应这些事件并执行相应的操作。
1. 引入Spring Cloud Bus依赖
首先,在需要使用Spring Cloud Bus的微服务项目中,添加Spring Cloud Bus的依赖。例如,在Maven项目中,可以在pom.xml
文件中添加如下依赖:
<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-bus-amqp</artifactId>
</dependency>
这里使用的是spring-cloud-starter-bus-amqp
,它依赖于Spring AMQP和RabbitMQ作为消息代理。如果你使用的是Kafka或其他消息代理,需要引入相应的starter依赖。
2. 配置消息代理
接下来,配置消息代理的连接信息。在application.yml
或application.properties
文件中添加相关配置:
spring:rabbitmq:host: localhostport: 5672username: guestpassword: guest
上述配置是针对RabbitMQ的,如果是其他消息代理,则需要按照相应的方式进行配置。
3. 发送消息
在服务中,你可以通过注入ApplicationEventPublisher
来发布事件,Spring Cloud Bus会自动将这些事件广播到所有微服务实例。
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationEventPublisher;
import org.springframework.stereotype.Service;@Service
public class BusService {@Autowiredprivate ApplicationEventPublisher applicationEventPublisher;public void sendMessage(final String message) {applicationEventPublisher.publishEvent(new CustomEvent(message));}
}
在上述代码中,CustomEvent
应该是ApplicationEvent
的一个子类,用于封装你想要传播的消息内容。
4. 监听消息
在需要监听消息的微服务中,使用@EventListener
注解来标记方法,该方法会在接收到指定类型的事件时被调用。
import org.springframework.context.event.EventListener;
import org.springframework.stereotype.Component;@Component
public class BusListener {@EventListenerpublic void handleCustomEvent(CustomEvent event) {// 处理接收到的消息System.out.println("Received custom event: " + event.getMessage());}
}
5. 刷新配置
Spring Cloud Bus的一个常见用法是刷新配置。当配置中心(如Spring Cloud Config Server)中的配置发生变化时,可以通过Spring Cloud Bus来通知所有微服务实例刷新它们的配置。
在微服务中,可以注入RefreshEndpoint
,并调用其refresh()
方法来刷新配置。
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cloud.context.refresh.RefreshEndpoint;
import org.springframework.stereotype.Service;@Service
public class ConfigRefreshService {@Autowiredprivate RefreshEndpoint refreshEndpoint;public void refreshConfigs() {refreshEndpoint.refresh();}
}
然后,你可以通过发送一个特定的消息来触发所有服务的配置刷新。这通常是通过HTTP请求到Spring Cloud Config Server来完成的,Config Server再通过Spring Cloud Bus来通知所有微服务。
注意事项
- 确保所有微服务都连接到了相同的消息代理,并且配置了相同的交换器和队列。
- 当使用Spring Cloud Bus进行配置刷新时,需要确保微服务有权限访问配置服务器,并且配置服务器已经启用了刷新端点。
- 根据业务需要,可以自定义事件类型和事件监听逻辑。
通过整合Spring Cloud Bus,微服务架构可以更加灵活地响应事件和状态变化,提高系统的响应速度和可靠性。
总结
通过本文的介绍,我们了解了事件驱动架构在微服务中的应用需求,以及如何通过Spring Cloud Stream和Spring Cloud Bus实现事件驱动微服务的集成。这些技术使得微服务之间的通信变得更加灵活和高效,为构建高可扩展性、高可靠性的分布式系统提供了有力支持。希望本文能够帮助读者深入理解并掌握Spring Cloud在事件驱动架构中的应用。
相关文章:

快速上手Spring Cloud 七:事件驱动架构与Spring Cloud
快速上手Spring Cloud 一:Spring Cloud 简介 快速上手Spring Cloud 二:核心组件解析 快速上手Spring Cloud 三:API网关深入探索与实战应用 快速上手Spring Cloud 四:微服务治理与安全 快速上手Spring Cloud 五:Spring …...
leetcode 1997.访问完所有房间的第一天
思路:动态规划前缀和 这道题还是很难的,因为你如果需要推出状态方程是很难想的。 在题中我们其实可以发现,这里在访问nextVisit数组的过程中,其实就是对于当前访问的房子之前的房子进行了回访。 怎么说呢?比如你现在…...

【InternLM 实战营第二期笔记】书生·浦语大模型全链路开源体系及InternLM2技术报告笔记
大模型 大模型成为发展通用人工智能的重要途径 专用模型:针对特定任务,一个模型解决一个问题 通用大模型:一个模型应对多种任务、多种模态 书生浦语大模型开源历程 2023.6.7:InternLM千亿参数语言大模型发布 2023.7.6&#…...

Netty对Channel事件的处理以及空轮询Bug的解决
继续上一篇Netty文章,这篇文章主要分析Netty对Channel事件的处理以及空轮询Bug的解决 当Netty中采用循环处理事件和提交的任务时 由于此时我在客户端建立连接,此时服务端没有提交任何任务 此时select方法让Selector进入无休止的阻塞等待 此时selectCnt进…...
【PostgreSQL】- 1.1 在 Debian 12 上安装 PostgreSQL 15
官方说明参考 (原文 PostgreSQL:Linux 下载 (Debian)) 默认情况下,PostgreSQL 在所有 Debian 版本中都可用。但是, Debians 的稳定版本“快照”了特定版本的 PostgreSQL 然后在该 Debian 版本的…...

第4章.精通标准提示,引领ChatGPT精准输出
标准提示 标准提示,是引导ChatGPT输出的一个简单方法,它提供了一个具体的任务让模型完成。 如果你要生成一篇新闻摘要。你只要发送指示词:汇总这篇新闻 : …… 提示公式:生成[任务] 生成新闻文章的摘要: 任务&#x…...

HTTP状态 405 - 方法不允许
方法有问题。 用Post发的请求,然后用Put接收的。 大家也可以看看是不是有这种问题 <body><h1>HTTP状态 405 - 方法不允许</h1><hr class"line" /><p><b>类型</b> 状态报告</p><p><b>消息…...

题目 2898: 二维数组回形遍历
题目描述: 给定一个row行col列的整数数组array,要求从array[0][0]元素开始,按回形从外向内顺时针顺序遍历整个数组。如图所示: 代码: package lanqiao;import java.math.BigInteger; import java.util.*;public class Main {public static …...

Git命令上传本地项目至github
记录如何创建个人仓库并上传已有代码至github in MacOS环境 0. 首先下载git 方法很多 这里就不介绍了 1. Github Create a new repository 先在github上创建一个空仓库,用于一会儿链接项目文件,按照自己的需求设置name和是否private 2.push an exis…...

机器学习之决策树现成的模型使用
目录 须知 DecisionTreeClassifier sklearn.tree.plot_tree cost_complexity_pruning_path(X_train, y_train) CART分类树算法 基尼指数 分类树的构建思想 对于离散的数据 对于连续值 剪枝策略 剪枝是什么 剪枝的分类 预剪枝 后剪枝 后剪枝策略体现之威斯康辛州乳…...

【python分析实战】成本:揭示电商平台月度开支与成本结构占比 - 过于详细 【收藏】
重点关注本文思路,用python分析,方便大家实验复现,代码每次都用全量的,其他工具自行选择。 全文3000字,阅读10min,操作1小时 企业案例实战欢迎关注专栏 每日更新:https://blog.csdn.net/cciehl/…...

新网站收录时间是多久,新建网站多久被百度收录
对于新建的网站而言,被搜索引擎收录是非常重要的一步,它标志着网站的正式上线和对外开放。然而,新网站被搜索引擎收录需要一定的时间,而且时间长短受多种因素影响。本文将探讨新网站收录需要多长时间,以及新建网站多久…...

通过Caliper进行压力测试程序,且汇总压力测试问题解决
环境要求 第一步. 配置基本环境 部署Caliper的计算机需要有外网权限;操作系统版本需要满足以下要求:Ubuntu >= 16.04、CentOS >= 7或MacOS >= 10.14;部署Caliper的计算机需要安装有以下软件:python 2.7、make、g++(gcc-c++)、gcc及git。第二步. 安装NodeJS # …...

LabVIEW比例流量阀自动测试系统
LabVIEW比例流量阀自动测试系统 开发了一套基于LabVIEW编程和PLC控制的比例流量阀自动测试系统。通过引入改进的FCMAC算法至测试回路的压力控制系统,有效提升了压力控制效果,展现了系统的设计理念和实现方法。 项目背景: 比例流量阀在液压…...

安卓U3D逆向从Assembly-CSharp到il2cpp
随着unity技术的发展及厂商对于脚本源码的保护,很大一部分U3D应用的scripting backend已经由mono转为了il2cpp,本文从unity简单应用的制作讲起,介绍U3D应用脚本的Assembly-CSharp.dll的逆向及il2cpp.so的逆向分析。 目录如下: 0…...

计算机网络——30SDN控制平面
SDN控制平面 SDN架构 数据平面交换机 快速、简单,商业化交换设备采用硬件实现通用转发功能流表被控制器计算和安装基于南向API,SDN控制器访问基于流的交换机 定义了哪些可以被控制哪些不能 也定义了和控制器的协议 SDN控制器(网络OS&#…...

Obsidian插件-高亮块(Admonition)
在插件市场里面搜索Admonition并安装插件,就可以使用高亮块了。 添加高亮块 用法稍微有一些不同。按照下面的格式,输入Markdown就可以创建一个高亮块。 内容内容内容输入*ad-*会出现相应的类型可以选择...
jHipster 之 webflux-前端用EventSource处理sse变成了批量处理而非实时处理
现象: const eventSource new EventSource(API_URL5);eventSource.onmessage streamEvent > {console.log(a message is come in--------->);const content streamEvent.data;console.log(Received content: content);};前端用EventSource 处理webflux的…...

原型链-(前端面试 2024 版)
来讲一讲原型链 原型链只存在于函数之中 四个规则 1、引用类型,都具有对象特性,即可自由扩展属性。 2、引用类型,都有一个隐式原型 __proto__ 属性,属性值是一个普通的对象。 3、引用类型,隐式原型 __proto__ 的属…...

网络套接字补充——UDP网络编程
五、UDP网络编程 1.对于服务器使用智能指针维护生命周期;2.创建UDP套接字;3.绑定端口号,包括设置服务器端口号和IP地址,端口号一般是2字节使用uint16_t,而IP地址用户习惯使用点分十进制格式所以传入的是string类型…...

日语AI面试高效通关秘籍:专业解读与青柚面试智能助攻
在如今就业市场竞争日益激烈的背景下,越来越多的求职者将目光投向了日本及中日双语岗位。但是,一场日语面试往往让许多人感到步履维艰。你是否也曾因为面试官抛出的“刁钻问题”而心生畏惧?面对生疏的日语交流环境,即便提前恶补了…...
【Linux】shell脚本忽略错误继续执行
在 shell 脚本中,可以使用 set -e 命令来设置脚本在遇到错误时退出执行。如果你希望脚本忽略错误并继续执行,可以在脚本开头添加 set e 命令来取消该设置。 举例1 #!/bin/bash# 取消 set -e 的设置 set e# 执行命令,并忽略错误 rm somefile…...

微软PowerBI考试 PL300-选择 Power BI 模型框架【附练习数据】
微软PowerBI考试 PL300-选择 Power BI 模型框架 20 多年来,Microsoft 持续对企业商业智能 (BI) 进行大量投资。 Azure Analysis Services (AAS) 和 SQL Server Analysis Services (SSAS) 基于无数企业使用的成熟的 BI 数据建模技术。 同样的技术也是 Power BI 数据…...

SpringBoot+uniapp 的 Champion 俱乐部微信小程序设计与实现,论文初版实现
摘要 本论文旨在设计并实现基于 SpringBoot 和 uniapp 的 Champion 俱乐部微信小程序,以满足俱乐部线上活动推广、会员管理、社交互动等需求。通过 SpringBoot 搭建后端服务,提供稳定高效的数据处理与业务逻辑支持;利用 uniapp 实现跨平台前…...

【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...

12.找到字符串中所有字母异位词
🧠 题目解析 题目描述: 给定两个字符串 s 和 p,找出 s 中所有 p 的字母异位词的起始索引。 返回的答案以数组形式表示。 字母异位词定义: 若两个字符串包含的字符种类和出现次数完全相同,顺序无所谓,则互为…...

C++ 求圆面积的程序(Program to find area of a circle)
给定半径r,求圆的面积。圆的面积应精确到小数点后5位。 例子: 输入:r 5 输出:78.53982 解释:由于面积 PI * r * r 3.14159265358979323846 * 5 * 5 78.53982,因为我们只保留小数点后 5 位数字。 输…...

用机器学习破解新能源领域的“弃风”难题
音乐发烧友深有体会,玩音乐的本质就是玩电网。火电声音偏暖,水电偏冷,风电偏空旷。至于太阳能发的电,则略显朦胧和单薄。 不知你是否有感觉,近两年家里的音响声音越来越冷,听起来越来越单薄? —…...

三分算法与DeepSeek辅助证明是单峰函数
前置 单峰函数有唯一的最大值,最大值左侧的数值严格单调递增,最大值右侧的数值严格单调递减。 单谷函数有唯一的最小值,最小值左侧的数值严格单调递减,最小值右侧的数值严格单调递增。 三分的本质 三分和二分一样都是通过不断缩…...
省略号和可变参数模板
本文主要介绍如何展开可变参数的参数包 1.C语言的va_list展开可变参数 #include <iostream> #include <cstdarg>void printNumbers(int count, ...) {// 声明va_list类型的变量va_list args;// 使用va_start将可变参数写入变量argsva_start(args, count);for (in…...