【架构之路】微服务中常用的几种通信方式

news2025/1/9 2:05:28

2024年,计算机相关专业还值得选择吗?

579a429daf314744b995f37351b46548

强烈推荐

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站:人工智能

b004071ozy_05_amzn

引言

微服务架构由于其灵活性、高可扩展性和易维护性,已成为构建复杂系统的主流选择。

微服务架构将系统拆分为多个独立的服务,每个服务负责特定的功能,并通过各种通信方式进行协作。

这些通信方式在确保系统高效、可靠运行的过程中起着至关重要的作用。

本文将介绍几种常见的微服务通信方式,包括HTTP REST、gRPC、消息队列和WebSocket,并通过Java示例说明它们的应用场景和实现方法。


几种通信方式

在微服务架构中,服务之间的通信是关键组件之一。常见的通信方式包括HTTP REST、gRPC、消息队列、以及基于WebSocket的通信。下面举例说明这些通信方式。

1. HTTP REST

HTTP REST是一种广泛使用的同步通信方式。每个微服务通过HTTP请求相互通信,通常使用JSON作为数据格式。

示例:

假设有两个微服务,一个用于用户管理(User Service),另一个用于订单管理(Order Service)。Order Service需要从User Service获取用户信息。

User Service:

@RestController
@RequestMapping("/users")
public class UserController {

    @GetMapping("/{id}")
    public ResponseEntity<User> getUserById(@PathVariable String id) {
        // 假设从数据库获取用户信息
        User user = userService.findUserById(id);
        return ResponseEntity.ok(user);
    }
}

Order Service:

@Service
public class UserServiceClient {

    private final RestTemplate restTemplate;

    @Autowired
    public UserServiceClient(RestTemplate restTemplate) {
        this.restTemplate = restTemplate;
    }

    public User getUserById(String userId) {
        String url = "http://USER-SERVICE/users/" + userId;
        return restTemplate.getForObject(url, User.class);
    }
}

配置RestTemplate:

@Configuration
public class RestTemplateConfig {

    @Bean
    public RestTemplate restTemplate() {
        return new RestTemplate();
    }
}
2. gRPC

gRPC是Google开源的一个高性能、通用的RPC框架,使用Protocol Buffers作为接口定义语言,并支持多种编程语言。

示例:

假设有两个微服务,一个用于用户管理(User Service),另一个用于订单管理(Order Service)。Order Service需要从User Service获取用户信息。

User Service:

1.定义.proto文件:

syntax = "proto3";

option java_package = "com.example.userservice";
option java_outer_classname = "UserServiceProto";

service UserService {
    rpc GetUser (UserRequest) returns (UserResponse) {}
}

message UserRequest {
    string id = 1;
}

message UserResponse {
    string id = 1;
    string name = 2;
    string email = 3;
}

2.实现服务端:

public class UserServiceImpl extends UserServiceGrpc.UserServiceImplBase {

    @Override
    public void getUser(UserRequest request, StreamObserver<UserResponse> responseObserver) {
        // 假设从数据库获取用户信息
        UserResponse response = UserResponse.newBuilder()
                .setId(request.getId())
                .setName("John Doe")
                .setEmail("john.doe@example.com")
                .build();
        responseObserver.onNext(response);
        responseObserver.onCompleted();
    }
}

3.配置并启动gRPC服务器:

public class GrpcServer {

    public static void main(String[] args) throws IOException, InterruptedException {
        Server server = ServerBuilder.forPort(8080)
                .addService(new UserServiceImpl())
                .build();

        server.start();
        System.out.println("Server started on port 8080");
        server.awaitTermination();
    }
}

Order Service:

1.创建gRPC客户端:

public class UserServiceClient {

    private final UserServiceGrpc.UserServiceBlockingStub userServiceStub;

    public UserServiceClient() {
        ManagedChannel channel = ManagedChannelBuilder.forAddress("localhost", 8080)
                .usePlaintext()
                .build();
        userServiceStub = UserServiceGrpc.newBlockingStub(channel);
    }

    public UserResponse getUserById(String userId) {
        UserRequest request = UserRequest.newBuilder().setId(userId).build();
        return userServiceStub.getUser(request);
    }
}
3. 消息队列

消息队列是一种异步通信方式,常用的消息队列系统有RabbitMQ、Apache Kafka等。消息队列可以解耦生产者和消费者,实现异步处理。

示例:

假设有两个微服务,一个用于订单管理(Order Service),另一个用于通知服务(Notification Service)。订单服务在订单创建后发送消息到消息队列,通知服务接收并处理消息。

Order Service:

@Service
public class OrderService {

    private final RabbitTemplate rabbitTemplate;

    @Autowired
    public OrderService(RabbitTemplate rabbitTemplate) {
        this.rabbitTemplate = rabbitTemplate;
    }

    public void createOrder(Order order) {
        // 创建订单逻辑
        rabbitTemplate.convertAndSend("order.exchange", "order.created", order);
    }
}

Notification Service:

@Service
public class NotificationService {

    @RabbitListener(queues = "order.queue")
    public void handleOrderCreated(Order order) {
        // 处理订单创建通知
        System.out.println("Received order: " + order);
    }
}

配置RabbitMQ:

@Configuration
public class RabbitMQConfig {

    @Bean
    public Queue queue() {
        return new Queue("order.queue");
    }

    @Bean
    public TopicExchange exchange() {
        return new TopicExchange("order.exchange");
    }

    @Bean
    public Binding binding(Queue queue, TopicExchange exchange) {
        return BindingBuilder.bind(queue).to(exchange).with("order.created");
    }
}

4. WebSocket

WebSocket是一种双向通信协议,适用于需要实时通信的场景。

示例:

假设有一个聊天应用,两个微服务分别处理用户和聊天信息。

Chat Service:

@Configuration
@EnableWebSocket
public class WebSocketConfig implements WebSocketConfigurer {

    @Override
    public void registerWebSocketHandlers(WebSocketHandlerRegistry registry) {
        registry.addHandler(new ChatWebSocketHandler(), "/chat");
    }
}

@Component
public class ChatWebSocketHandler extends TextWebSocketHandler {

    @Override
    public void handleTextMessage(WebSocketSession session, TextMessage message) throws Exception {
        // 处理收到的消息
        session.sendMessage(new TextMessage("Received: " + message.getPayload()));
    }
}

应用场景

不同的通信方式适用于不同的应用场景,每种方式都有其优缺点和适用领域。以下是对上述几种通信方式的应用场景的说明:

1. HTTP REST

应用场景:

  • Web服务与API接口:HTTP REST是构建Web服务和API接口的首选方法,广泛应用于提供对外部系统的访问接口。
  • 同步请求响应:适用于需要立即得到响应的请求,比如用户查询、订单查询等。
  • 简单易用:对开发者友好,易于实现和调试,适合快速开发。

示例场景:

  • 用户注册、登录等操作。

  • 产品信息查询,订单管理系统。

2. gRPC

应用场景:

  • 高性能通信:适用于需要高性能、低延迟通信的场景,如微服务之间的大量数据传输。
  • 多语言支持:适用于多语言环境,因为gRPC支持多种编程语言。
  • 严格的接口定义:适用于需要严格接口和数据类型约束的场景,通过Protocol Buffers定义接口。

示例场景:

  • 实时数据处理,如在线游戏、实时数据分析。

  • 微服务内部通信,如电商系统中的订单服务与库存服务之间的通信。

3. 消息队列

应用场景:

  • 异步处理:适用于需要异步处理的场景,避免长时间的同步等待。
  • 解耦系统:适用于希望解耦生产者和消费者的场景,使得系统更加灵活和可扩展。
  • 任务队列:适用于需要将任务放入队列中逐步处理的场景,如邮件发送、日志处理。

示例场景:

  • 订单创建后发送通知或进行库存更新。

  • 用户注册后发送欢迎邮件。

  • 日志收集与处理系统。

4. WebSocket

应用场景:

  • 实时通信:适用于需要实时双向通信的场景,如聊天应用、在线游戏、实时协作工具。
  • 低延迟要求:适用于对延迟有严格要求的应用,能够提供持续的低延迟连接。
  • 状态保持:适用于需要保持连接状态的应用,如实时数据更新。

示例场景:

  • 聊天应用,如即时消息系统。

  • 实时交易平台,如股票交易、加密货币交易。

  • 实时协作工具,如在线文档协作、实时编辑器。

场景应用总结
  • HTTP REST适用于简单的请求响应模型和对外提供API的场景,易于实现和使用。
  • gRPC适用于需要高性能通信和严格接口定义的场景,适合多语言环境和实时数据处理。
  • 消息队列适用于异步处理和解耦的场景,适合任务队列和事件驱动架构。
  • WebSocket适用于需要实时双向通信和低延迟的场景,适合实时应用和需要保持连接状态的场景。

根据具体的业务需求和性能要求,开发者可以选择最适合的通信方式来实现微服务之间的通信。


总结

微服务通信方式的选择在很大程度上影响系统的性能、可靠性和扩展性。

通过了解和掌握HTTP REST、gRPC、消息队列和WebSocket等不同通信方式的特点和应用场景,开发者可以根据具体业务需求选择最合适的通信方式,从而构建出高效、灵活和可扩展的微服务系统。

在实际应用中,可能需要结合多种通信方式,以发挥各自的优势,满足系统的不同需求。


强烈推荐

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站:人工智能

b004071ozy_05_amzn


专栏集锦

大佬们可以收藏以备不时之需:

Spring Boot 专栏:http://t.csdnimg.cn/peKde

ChatGPT 专栏:http://t.csdnimg.cn/cU0na

Java 专栏:http://t.csdnimg.cn/YUz5e

Go 专栏:http://t.csdnimg.cn/Jfryo

Netty 专栏:http://t.csdnimg.cn/0Mp1H

Redis 专栏:http://t.csdnimg.cn/JuTue

Mysql 专栏:http://t.csdnimg.cn/p1zU9

架构之路 专栏:http://t.csdnimg.cn/bXAPS


写在最后

感谢您的支持和鼓励! 😊🙏

如果大家对相关文章感兴趣,可以关注公众号"架构殿堂",会持续更新AIGC,java基础面试题, netty, spring boot, spring cloud等系列文章,一系列干货随时送达!

如果有项目或者毕设合作,请V:fengyelin8866,备注毕设

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1814263.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

mysql打开远程访问

这里写目录标题 1.使用navicat进入命令控制板,进入use mysql;2.查询用户表3.更新user表中root用户域属性&#xff0c;%表示允许外部访问4.执行以上语句之后再执行&#xff0c;FLUSH PRIVILEGES;5. 执行授权语句 1.使用navicat进入命令控制板,进入use mysql; use mysql;2.查询用…

NOSQL -- MOGODB

Mogodb简介: 是一个开源的, 高性能, 无模式的文档型数据库. NoSql数据库产品当中的一种, 也是最像关系型数据库的非关系型数据库 使用场景: 针对不同的应用场景, 以及其对应的修改对应数据的频率, 我们可以以此选择需要哪一种类型的数据库 Mongo的使用: 启动: 在解压完成之后…

动手学操作系统(六、获取物理内存容量)

动手学操作系统&#xff08;六、获取物理内存容量&#xff09; 在上一节中&#xff0c;我们介绍了保护模式和实模式的区别&#xff0c;保护模式的最大特点是“大”&#xff0c;“大”是指寻址空间大&#xff0c;在进入保护模式之后&#xff0c;我们还将要接触虚拟内存、内存管…

基于pytorch实现的DenseUnet医学图像分割(腹部多脏器)

1、前言 本章将介绍将densenet的主干网络引入unet中 官方实现的代码&#xff1a;kits19-challenge/network at master nitsaick/kits19-challenge (github.com) 本章实现的项目目录如下&#xff1a; 主要代码有train、evaluate、predict脚本 2、代码介绍 数据预处理脚本 数据…

全能型施耐德可编程控制器M241介绍

施耐德M241是一款通信强大、定位控制、丰富扩展于一身的全能型可编程控制器&#xff0c;适用于具有速度控制和位置控制功能的高性能一体型设备。其内置以太网通信端口&#xff0c;可以提供FTP和网络服务器功能&#xff0c;能够更为便捷地整合到控制系统架构中&#xff0c;通过智…

【0基础学爬虫】爬虫基础之自动化工具 DrissionPage 的使用

概述 前三期文章中已经介绍到了 Selenium 与 Playwright 、Pyppeteer 的使用方法&#xff0c;它们的功能都非常强大。而本期要讲的 DrissionPage 更为独特&#xff0c;强大&#xff0c;而且使用更为方便&#xff0c;目前检测少&#xff0c;强烈推荐&#xff01;&#xff01;&a…

Spark Streaming 概述及入门案例

一、介绍 1. 不同的数据处理 从数据处理的方式&#xff1a; 流式数据处理(Streaming)批量数据处理(Batch) 从数据处理的延迟&#xff1a; 实时数据处理(毫秒级别)离线数据处理(小时或天级别) 2. 简介 SparkStreaming 是一个准实时(秒或分钟级别)、微批量的数据处理框架Spa…

了解侧信道攻击基础知识

人们通常认为特洛伊木马、恶意软件和其他形式的黑客攻击等漏洞是的威胁&#xff1b;然而&#xff0c;从 EE 的角度来看&#xff0c;安全性具有全新的含义。  事实上&#xff0c;许多的安全威胁都是基于硬件的&#xff0c;攻击者可以直接从运行我们的安全加密软件的硬件中窃取…

根据mooc 数据库旧代码 实现剥离数据库链接单独成类,并进行测试

数据源详情链接&#xff0c;SQLserver 2019 代码复制粘贴可产生数据 数据库JDBC 查询sqlserver 2019 利用模板实现输入查询-CSDN博客 效果如下 剥离的链接模块 Slinkv2.java package SQLadd;import java.sql.Connection; import java.sql.DriverManager; import java.sql.Re…

CentOS-内网搭建FTP-Server

一、镜像选择 1、 Centos-everting或者DVD 2、7.5 7.6 7.9 均可 二、安装步骤 1、其余步骤和普通安装一致。 2、最重要的一步为“软件选择” 1、勾选FTP、文件以及存储服务器、性能以及开发工具。 三、FTPServer搭建 1、关闭防火墙 systemctl stop firewalld or 通过21和20…

PS教程系统17

橡皮擦工具 主要配合画笔工具来使用 选择画笔工具新建图层试验擦除线条 如果直接在背景图片上进行擦除 会有背景颜色补充 背景橡皮擦 将其白色背景擦除掉shift相关键&#xff0c;进行工作区域切换吸取样点一次采样、两次采样连续、不连续等功能 在进行涂擦的过程一…

为中小制造企业注入数字化转型活力

随着劳动力成本上升,原材料价格上涨,企业生产成本逐年增加&#xff0c;市场竞争越来越激烈&#xff0c;传统的中小制造企业面临着巨大的压力。 通过数字化转型应对环境的变化已成为行业共识&#xff0c;在数字化的进程中&#xff0c;中小企业首要考虑生存问题&#xff0c;不能…

通过ClangFormat进行格式设置时出现错误

1、打开工具->选项 2、文本编辑器->C/C->格式设置->启动ClangFormat支持(去掉勾选)

六种图算法的python实现

六种图算法的python实现 1. Prim 算法 基本原理 Prim算法是一种求解最小生成树的贪心算法。所谓最小生成树&#xff0c;就是对于给定的连通图&#xff0c;找到一棵包含所有顶点的树&#xff0c;且树上所有边的权重之和最小。Prim算法从一个顶点开始&#xff0c;每次选择与当…

[CUDA编程] cuda graph优化心得

CUDA Graph 1. cuda graph的使用场景 cuda graph在一个kernel要多次执行&#xff0c;且每次只更改kernel 参数或者不更改参数时使用效果更加&#xff1b;但是如果将graph替换已有的kernel组合&#xff0c;且没有重复执行&#xff0c;感觉效率不是很高反而低于原始的kernel调用…

使用fvm切换flutter版本

切换flutter版本 下载fvm 1、dart pub global activate fvm dart下载fvm 2、warning中获取下载本地的地址 3、添加用户变量path&#xff1a; 下载地址 终端查看fvm版本 fvm --version 4、指定fvm文件缓存地址 fvm config --cache-path C:\src\fvm&#xff08;自定义地址&…

北京多商入驻app开发项目的主要优势及功能

多商入驻app开发项目的定义 随着电子支付技术的不断成熟&#xff0c;全国各地的消费者通过网络在线上购物的频率越来越高&#xff0c;为此&#xff0c;多商入驻app开发项目应用而生。各商家也纷纷开始申请入驻商城平台&#xff0c;开设自己的店铺。 图片来源&#xff1a;unspl…

【Ardiuno】实验使用ESP32单片机连接Wifi(图文)

ESP32单片机最为精华和有特色的地方当然是wifi连接&#xff0c;这里我们就写程序实验一下适使用ESP32主板连接wifi&#xff0c;为了简化实验我们这里只做了连接部分&#xff0c;其他实验在后续再继续。 由于本实验只要在串口监视器中查看结果状态即可&#xff0c;因此电路板上…

Ubuntu server 24 (Linux) lvm 动态扩容磁盘空间

1 查看磁盘信息 sudo fdisk -l 2 查看lvm分区信息 sudo lvdisplay 3 扩展逻辑卷 sudo lvextend -l 100%FREE /dev/ubuntu-vg/ubuntu-lv 4 刷新逻辑卷 sudo resize2fs /dev/ubuntu-vg/ubuntu-lv 5 查看磁盘信息 df -h

[Shell编程学习路线]——编制第一个shell脚本入门篇

&#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f6e0;️Shell编程专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2024年6月12日10点23分 &#x1f004;️文章质量&#xff1a;93分 目录 ——前言—— &#x1f4a5;常用的几种shell Bash Sh …