”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 微服务系统中服务之间的通信方式

微服务系统中服务之间的通信方式

发布于2024-08-21
浏览:814

1. 同步通讯

同步通信涉及实时交互,其中一个服务向另一个服务发送请求并暂停其操作,直到收到响应。 REST API 和 gRPC 是用于促进此类通信的常用协议。

Ways of communication between services in a Microservice system

1.1 REST API

RESTful API(表述性状态传输)是微服务系统中服务相互通信最常用的方法之一。 REST 利用 HTTP/HTTPS 和 JSON 或 XML 格式进行数据交换。

通常,服务通过直接调用另一个服务的 API 来相互交互。

请求和响应示例:

GET /users/12345 HTTP/1.1
Host: api.userservice.com
Accept: application/json
Authorization: Bearer your-access-token

{
  "userId": "12345",
  "name": "Michel J",
  "email": "[email protected]",
  "address": "Mountain View, Santa Clara, California"
}

源代码示例

import org.springframework.web.client.RestTemplate;
import org.springframework.http.ResponseEntity;

public class OrderService {

    private final RestTemplate restTemplate = new RestTemplate();
    private final String userServiceUrl = "https://api.userservice.com/users/";

    public User getUserById(String userId) {
        String url = userServiceUrl   userId;
        ResponseEntity response = restTemplate.getForEntity(url, User.class);
        return response.getBody();
    }
}

优点:

易于部署并与各种语言和工具集成。

能够轻松使用测试和监控工具。

缺点:

由于其同步特性,对于高速要求可能效率不高。

在处理网络错误或断开连接时可能会遇到困难。

1.2 通用RPC

gRPC,全称Google Remote procedure Call,是一个高性能、开源的通用RPC框架。它利用 HTTP/2 进行高效的数据传输,并且通常依赖于协议缓冲区(一种语言中立、平台中立、可扩展的机制,用于序列化结构化数据)来定义发送和接收的数据的结构。

示例,协议缓冲区的定义

syntax = "proto3";

package userservice;

// Define message User
message User {
    string userId = 1;
    string name = 2;
    string email = 3;
    string address = 4;
}

// Define service UserService
service UserService {
    rpc GetUserById (UserIdRequest) returns (User);
}

// Define message UserIdRequest
message UserIdRequest {
    string userId = 1;
}

对于用户管理服务,您应该实现一个遵循 .proto 文件中提供的服务定义的 gRPC 服务器。这包括创建必要的服务器端逻辑来处理传入的 gRPC 请求并生成适当的响应。

import io.grpc.stub.StreamObserver;
import userservice.User;
import userservice.UserIdRequest;
import userservice.UserServiceGrpc;

public class UserServiceImpl extends UserServiceGrpc.UserServiceImplBase {

    @Override
    public void getUserById(UserIdRequest request, StreamObserver responseObserver) {
        // Assuming you have a database to retrieve user information.
        User user = User.newBuilder()
                .setUserId(request.getUserId())
                .setName("Michel J")
                .setEmail("[email protected]")
                .setAddress("Mountain View, Santa Clara, California")
                .build();

        responseObserver.onNext(user);
        responseObserver.onCompleted();
    }
}

import io.grpc.Server;
import io.grpc.ServerBuilder;

public class UserServer {
    public static void main(String[] args) throws Exception {
        Server server = ServerBuilder.forPort(9090)
                .addService(new UserServiceImpl())
                .build()
                .start();

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

优点:

由于使用了 HTTP/2 和 Protocol Buffers,因此具有高性能和带宽效率。

支持多种编程语言,具有良好的扩展性。

缺点:

如果服务不支持 gRPC,则需要转换层。

部署和管理可能更加复杂。

2. 异步通信

异步通信是指一个服务向另一个服务发送请求而不阻塞其自身操作以等待回复的过程。这通常是通过消息队列或发布/订阅系统来实现的。

Ways of communication between services in a Microservice system

1.消息队列

消息队列系统,如 RabbitMQ 和 Apache ActiveMQ,促进服务之间的异步通信。

优点:

提高可扩展性和容错能力:系统可以更好地处理增加的工作负载并且不易出现故障。

减少服务负载:通过解耦请求发送和接收,主要服务可以专注于处理任务,而不会被不断的请求淹没。

缺点:

可能需要额外的工作来管理和维护:基于队列的系统可能更复杂,并且需要更多的资源来运行。

处理排序和确保消息传递的困难:确保以正确的顺序处理请求并且不丢失消息可能是一项技术挑战。

2.2.发布/订阅系统

Pub/Sub(发布/订阅)系统,例如 Apache Kafka 或 Google Pub/Sub,允许服务发布消息和订阅主题。

优点:

支持大规模和高吞吐量的数据流。

减少服务之间的依赖关系。

缺点:

需要更复杂的层来管理和监控主题和消息。

处理消息的排序和可靠性问题可能具有挑战性。”

如果您有兴趣,可以阅读我之前关于 pub/sub 主题的文章。

消息代理中的死信队列第 1 部分

消息代理中的死信队列第 2 部分

消息代理系统内的一致性和可靠性问题

3. 事件驱动的沟通

事件驱动的通信是指服务发出事件并且其他服务根据这些事件做出响应或采取操作。

3.1.同步事件

当服务发出事件并等待其他服务的响应时,就会发生同步事件。

优点:

易于控制和监控事件处理过程。

缺点:

如果响应服务缓慢或遇到错误,可能会导致瓶颈

3.2.异步事件

当服务发出事件并且不需要等待立即响应时,就会发生异步事件。

优点:

减少等待时间并提高可扩展性。

帮助服务更加独立运行并减少相互依赖。

缺点:

需要额外的机制来确保正确及时地处理事件。

难以确保顺序和处理重复事件。

4. 结论

微服务系统中服务间通信方式的选择取决于性能需求、可靠性、系统复杂度等因素。每种方法都有自己的优点和缺点,了解这些方法将帮助您构建更高效、更灵活的微服务系统。仔细考虑您系统的要求,选择最合适的通信方式。

版本声明 本文转载于:https://dev.to/anh_trntun_4732cf3d299/ways-of-communication-between-services-in-a-microservice-system-597p?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • eval()vs. ast.literal_eval():对于用户输入,哪个Python函数更安全?
    eval()vs. ast.literal_eval():对于用户输入,哪个Python函数更安全?
    称量()和ast.literal_eval()中的Python Security 在使用用户输入时,必须优先确保安全性。强大的Python功能Eval()通常是作为潜在解决方案而出现的,但担心其潜在风险。 This article delves into the differences betwee...
    编程 发布于2025-05-25
  • 如何在Java中正确显示“ DD/MM/YYYY HH:MM:SS.SS”格式的当前日期和时间?
    如何在Java中正确显示“ DD/MM/YYYY HH:MM:SS.SS”格式的当前日期和时间?
    如何在“ dd/mm/yyyy hh:mm:mm:ss.ss”格式“ gormat 解决方案:的,请访问量很大,并应为procectiquiestate的,并在整个代码上正确格式不多: java.text.simpledateformat; 导入java.util.calendar; 导入java...
    编程 发布于2025-05-25
  • C++中如何将独占指针作为函数或构造函数参数传递?
    C++中如何将独占指针作为函数或构造函数参数传递?
    在构造函数和函数中将唯一的指数管理为参数 unique pointers( unique_ptr [2启示。通过值: base(std :: simelor_ptr n) :next(std :: move(n)){} 此方法将唯一指针的所有权转移到函数/对象。指针的内容被移至功能中,在操作...
    编程 发布于2025-05-25
  • 如何高效地在一个事务中插入数据到多个MySQL表?
    如何高效地在一个事务中插入数据到多个MySQL表?
    mySQL插入到多个表中,该数据可能会产生意外的结果。虽然似乎有多个查询可以解决问题,但将从用户表的自动信息ID与配置文件表的手动用户ID相关联提出了挑战。使用Transactions和last_insert_id() 插入用户(用户名,密码)值('test','test...
    编程 发布于2025-05-25
  • 为什么HTML无法打印页码及解决方案
    为什么HTML无法打印页码及解决方案
    无法在html页面上打印页码? @page规则在@Media内部和外部都无济于事。 HTML:Customization:@page { margin: 10%; @top-center { font-family: sans-serif; font-weight: bo...
    编程 发布于2025-05-25
  • Spark DataFrame添加常量列的妙招
    Spark DataFrame添加常量列的妙招
    在Spark Dataframe ,将常数列添加到Spark DataFrame,该列具有适用于所有行的任意值的Spark DataFrame,可以通过多种方式实现。使用文字值(SPARK 1.3)在尝试提供直接值时,用于此问题时,旨在为此目的的column方法可能会导致错误。 df.withCo...
    编程 发布于2025-05-25
  • Java中Lambda表达式为何需要“final”或“有效final”变量?
    Java中Lambda表达式为何需要“final”或“有效final”变量?
    Lambda Expressions Require "Final" or "Effectively Final" VariablesThe error message "Variable used in lambda expression shou...
    编程 发布于2025-05-25
  • Java中如何使用观察者模式实现自定义事件?
    Java中如何使用观察者模式实现自定义事件?
    在Java 中创建自定义事件的自定义事件在许多编程场景中都是无关紧要的,使组件能够基于特定的触发器相互通信。本文旨在解决以下内容:问题语句我们如何在Java中实现自定义事件以促进基于特定事件的对象之间的交互,定义了管理订阅者的类界面。以下代码片段演示了如何使用观察者模式创建自定义事件: args)...
    编程 发布于2025-05-25
  • 如何将来自三个MySQL表的数据组合到新表中?
    如何将来自三个MySQL表的数据组合到新表中?
    mysql:从三个表和列的新表创建新表 答案:为了实现这一目标,您可以利用一个3-way Join。 选择p。*,d.content作为年龄 来自人为p的人 加入d.person_id = p.id上的d的详细信息 加入T.Id = d.detail_id的分类法 其中t.taxonomy =...
    编程 发布于2025-05-25
  • 如何在Java的全屏独家模式下处理用户输入?
    如何在Java的全屏独家模式下处理用户输入?
    Handling User Input in Full Screen Exclusive Mode in JavaIntroductionWhen running a Java application in full screen exclusive mode, the usual event ha...
    编程 发布于2025-05-25
  • 找到最大计数时,如何解决mySQL中的“组函数\”错误的“无效使用”?
    找到最大计数时,如何解决mySQL中的“组函数\”错误的“无效使用”?
    如何在mySQL中使用mySql 检索最大计数,您可能会遇到一个问题,您可能会在尝试使用以下命令:理解错误正确找到由名称列分组的值的最大计数,请使用以下修改后的查询: 计数(*)为c 来自EMP1 按名称组 c desc订购 限制1 查询说明 select语句提取名称列和每个名称...
    编程 发布于2025-05-25
  • 在JavaScript中如何并发运行异步操作并正确处理错误?
    在JavaScript中如何并发运行异步操作并正确处理错误?
    同意操作execution 在执行asynchronous操作时,相关的代码段落会遇到一个问题,当执行asynchronous操作:此实现在启动下一个操作之前依次等待每个操作的完成。要启用并发执行,需要进行修改的方法。 第一个解决方案试图通过获得每个操作的承诺来解决此问题,然后单独等待它们: co...
    编程 发布于2025-05-25
  • `console.log`显示修改后对象值异常的原因
    `console.log`显示修改后对象值异常的原因
    foo = [{id:1},{id:2},{id:3},{id:4},{id:id:5},],]; console.log('foo1',foo,foo.length); foo.splice(2,1); console.log('foo2', foo, foo....
    编程 发布于2025-05-25
  • 用户本地时间格式及时区偏移显示指南
    用户本地时间格式及时区偏移显示指南
    在用户的语言环境格式中显示日期/时间,并使用时间偏移在向最终用户展示日期和时间时,以其localzone and格式显示它们至关重要。这确保了不同地理位置的清晰度和无缝用户体验。以下是使用JavaScript实现此目的的方法。方法:推荐方法是处理客户端的Javascript中的日期/时间格式化和时...
    编程 发布于2025-05-25
  • 如何在php中使用卷发发送原始帖子请求?
    如何在php中使用卷发发送原始帖子请求?
    如何使用php 创建请求来发送原始帖子请求,开始使用curl_init()开始初始化curl session。然后,配置以下选项: curlopt_url:请求 [要发送的原始数据指定内容类型,为原始的帖子请求指定身体的内容类型很重要。在这种情况下,它是文本/平原。要执行此操作,请使用包含以下标头...
    编程 发布于2025-05-25

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3