网络套件字(理论知识)

一、IP地址和目的IP地址

上次说到IP地址是为了是为了让信息正确的从原主机传送到目的主机,而原IP地址和目的IP地址就是用于标识两个主机的,既然叫做地址必然有着路径规划的作用,而路径规划最重要的就是,从哪来到哪去,现在在哪下一步去哪?

比如我要从山西北部骑行到江苏南京,那么我的源地址就是山西,目的地就是南京,第一步山西->河北:第二步河北->山东;第三步山东->江苏;

从上面我们看到的一个简单的路径规划,可以看到,出发地和目的地是不变的而当前位置和下一步的位置是一直在变化的,而IP数据包头部中, 有两个IP地址, 分别叫做源IP地址, 和目的IP地址。

在数据进行传输之前,会先自顶向下贯穿网络协议栈完成数据的封装(每一层会根据对应的协议添加报头),其中在网络层封装的IP报头当中就涵盖了源IP地址和目的IP地址。而除了源IP地址和目的IP地址之外,还有源MAC地址和目的MAC地址的概念。

二、理解源MAC地址和目的MAC地址

2.1MAC地址

MAC地址(Media Access Control Address)是一个用于识别网络设备的唯一标识符。每个网络设备(如计算机、手机、路由器等)都有一个独特的MAC地址。MAC地址通常是由48位二进制数字组成,通常以十六进制表示。MAC地址由厂商在生产设备时分配,分为两部分:前24位是厂商标识符,后24位是设备标识符。MAC地址在数据链路层(OSI模型中的第二层)使用,用于在局域网中唯一标识设备。MAC地址的作用类似于身份证号码,用于在网络中确定设备的身份和位置。

 通常数据的传输是跨局域网的,数据在传输过程中会经过若干个路由器,

而在上篇博客中提到路由器是看作在TCP/IP五层(或四层)模型中的网络层。

而当数据在局域网中传输时,就需要使用到数据链路层,而在该层要使用的就是MAC地址。

2.2源MAC地址和目的MAC地址

当数据在局域网中传输时,数据帧会包含发送者和接收者的MAC地址。网络设备根据目标MAC地址来决定是否接收该数据帧,如果目标MAC地址与自身的MAC地址匹配,设备就会接收并处理该数据帧。

 上面提到的发送者和接收者的MAC地址其实就是源MAC地址和目的MAC地址。

源MAC地址和目的MAC地址是包含在链路层的报头当中的,而MAC地址实际只在当前局域网内有效,因此当数据跨网络到达另一个局域网时,其源MAC地址和目的MAC地址就需要发生变化,因此当数据达到路由器时,路由器会将该数据当中链路层的报头去掉,然后再重新封装一个报头,此时该数据的源MAC地址和目的MAC地址就发生了变化。

这样做就可以让局域网技术不同的局域网之间经行通信,就像上图一样即使局域网的技术不同(一个是以太网,一个是令牌环网)但是数据是在进入局域网的时候才会添加数据链路层的报头,如下图。

 

2.3数据传输的两套地址

  • 一套是源IP地址和目的IP地址,这两个地址在数据传输过程中基本是不会发生变化的(存在一些特殊情况,比如在数据传输过程中使用NET技术,其源IP地址会发生变化,但至少目的IP地址是不会变化的)。
  • 另一套就是源MAC地址和目的MAC地址,这两个地址是一直在发生变化的,因为在数据传输的过程中路由器不断在进行解包和重新封装。

三、端口号

知道了消息如何在两台不同的主机之间传递,那么当消息传递到另一台主机后,如何知道该消息是发送给主机上哪一个应用呢?

比如QQ之间进行通讯,,可以看作是两个不同主机之间进程之间的通讯,主机与主机之间通过ip地址不走错,而进程带有一个端口号,每个主机都有独一无二的IP而一台主机上的每个进程都有唯一的端口号。

端口号 (port) 是传输层协议的内容
  • 端口号是一个2字节16位的整数;
  • 端口号用来标识一个进程, 告诉操作系统, 当前的这个数据要交给哪一个进程来处理;
  • IP地址 + 端口号能够标识网络上的某一台主机的某一个进程;
  • 一个端口号只能被一个进程占用

既然提到了进程我们都知道进程pid是用于标识唯一进程的那么他们有什么关系呢?

虽然进程PID和端口号都是用于唯一标识某种资源(进程或网络服务),但它们之间并没有直接的关联。在实际的网络通信中,操作系统会维护一个端口号与进程之间的映射关系,使得特定端口号的数据能够被正确路由到相应的进程。通常,网络服务启动时会绑定到一个特定的端口号,并且在运行期间会监听该端口,从而等待传入的连接请求或数据包。

传输层协议(TCP和UDP)的数据段中有两个端口号, 分别叫做源端口号和目的端口号. 就是在描述 "数据是谁发的, 要发给谁";
综上 网络通信的本质就是进程间经行通信

四、浅谈UDP/TCP

在前面我们简单谈了在数据链路层(MAC)和传输层(IP)中十分重要的概念,通过这两层,我们已经能将数据从一台主机传输到另一台主机。但是数据的安全性无法保证,而数据;链路层就是用于为应用层提供可靠的、端到端的数据传输服务,隐藏了网络通信的细节,使得应用程序能够简单地进行数据交换而不需要关心底层网络的细节。传输层的核心协议包括TCP和UDP,它们提供了不同级别的服务,满足不同应用场景的需求。

3.1UDP

UDP(User Datagram Protocol)用户数据报协议是一种无连接的、不可靠的传输协议,属于网络通信协议簇的一部分。与TCP不同,UDP不提供可靠的数据传输和错误恢复机制,而是专注于在网络上传输数据包,提供简单的数据传输服务。

 以下是UDP的一些特点:

  1. 无连接性(Connectionless):UDP是一种无连接的协议,发送数据之前不需要建立连接,也不维护连接状态。

  2. 不可靠性(Unreliable):UDP不提供数据包的可靠传输,数据包可能会丢失、重复或顺序错乱。

  3. 轻量级(Lightweight):UDP的头部开销比TCP小,不需要维护连接状态,因此具有较低的延迟。

  4. 面向数据报(Datagram-Oriented):UDP以数据报为单位进行数据传输,每个数据报独立于其他数据报。

  5. 不提供拥塞控制(No Congestion Control):UDP不提供拥塞控制机制,数据包可能会因为网络拥塞而丢失。

注意 上面的不可靠性只是一个相对的概念并不是说他不可靠就一定数据会出错。

UDP常用于对实时性要求较高、数据量较小、传输延迟较低的应用场景,例如音频和视频流传输、DNS查询、实时游戏等。由于其简单和高效的特性,UDP在一些特定的网络应用中具有重要的作用。比如我们经常玩的王者荣耀和原神都是udp链接

3.2TCP

TCP(Transmission Control Protocol)是一种面向连接的、可靠的、基于字节流的传输层协议,是互联网中最常用的协议之一。它提供了可靠的数据传输和错误恢复机制,用于在网络中进行端到端的数据传输。

 以下是TCP的一些主要特点:

  1. 面向连接(Connection-Oriented):在数据传输之前,TCP需要先建立连接,然后再进行数据传输,传输完成后再关闭连接。这种连接的建立和释放过程确保了数据的可靠传输。

  2. 可靠性(Reliability):TCP通过使用序号、确认和重传机制来确保数据的可靠传输。接收方会确认接收到的数据,并在需要时请求重传丢失的数据,从而保证数据的完整性和按序传输。

  3. 流量控制(Flow Control):TCP使用滑动窗口机制进行流量控制,确保发送方和接收方之间的数据传输速率匹配,防止数据发送过快导致接收方缓冲区溢出。

  4. 拥塞控制(Congestion Control):TCP使用拥塞窗口和拥塞避免机制来进行拥塞控制,防止网络拥塞导致的数据丢失和传输延迟增加。

  5. 面向字节流(Byte-Oriented):TCP是一种面向字节流的协议,它不保留消息的边界,而是将数据视为一连串的字节流进行传输。

  6. 全双工通信(Full Duplex Communication):TCP连接是全双工的,允许双方同时发送和接收数据。

TCP被广泛用于诸如网页浏览、文件传输、电子邮件和远程登录等应用中,它的可靠性和稳定性使得它成为互联网通信的重要基础。 

五、socket

在套接字编程中,常常将IP地址和端口号结合起来表示一个通信的端点,这种组合称为套接字地址。因此,可以说IP地址和端口号一起构成了一个套接字地址。然而,严格来说,套接字是操作系统中的一个抽象概念,用于表示网络通信的端点,而IP地址和端口号只是套接字地址的组成部分,用于确定通信的目的地或来源。因此,套接字通常是由IP地址、端口号和协议类型(如TCP或UDP)一起确定的.

5.1socket编程接口

在C语言中,使用套接字(socket)进行网络编程时,常见的编程接口包括:

 socket(): 创建一个套接字,返回套接字描述符。

int socket(int domain, int type, int protocol);
  • domain: 地址族,如 AF_INET(IPv4)或 AF_INET6(IPv6)。
  • type: 套接字类型,如 SOCK_STREAM(流套接字,TCP)或 SOCK_DGRAM(数据报套接字,UDP)。
  • protocol: 协议类型,通常为 0,表示由系统自动选择。

bind(): 将套接字与特定的IP地址和端口号绑定。

int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
  • sockfd: 套接字描述符。
  • addr: 指向要绑定的 sockaddr 结构体的指针。
  • addrlen: 地址结构体的大小。

listen(): 在服务器端开始监听连接请求。

int listen(int sockfd, int backlog);
  • sockfd: 套接字描述符。
  • backlog: 允许的连接等待队列的最大长度。

accept(): 接受客户端的连接请求,并创建一个新的套接字用于与客户端进行通信。

int accept(int sockfd, struct sockaddr *addr, socklen_t *addrlen);
  • sockfd: 服务器套接字描述符。
  • addr: 用于存储客户端地址信息的 sockaddr 结构体。
  • addrlen: 指向存储客户端地址长度的变量的指针。

connect(): 连接到服务器。

int connect(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
  • sockfd: 套接字描述符。
  • addr: 服务器地址的 sockaddr 结构体指针。
  • addrlen: 地址结构体的大小。

send() / recv(): 发送和接收数据。

ssize_t send(int sockfd, const void *buf, size_t len, int flags);
  • sockfd: 套接字描述符。
  • buf: 要发送的数据的缓冲区。
  • len: 要发送的数据的字节数。
  • flags: 发送标志,通常为 0
ssize_t recv(int sockfd, void *buf, size_t len, int flags);
  • sockfd: 套接字描述符。
  • buf: 接收数据的缓冲区。
  • len: 缓冲区长度。
  • flags: 接收标志,通常为 0

sendto() / recvfrom(): 用于在无连接的套接字上发送和接收数据报

ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,const struct sockaddr *dest_addr, socklen_t addrlen);
  • sockfd: 套接字描述符。
  • buf: 要发送的数据的缓冲区。
  • len: 要发送的数据的字节数。
  • flags: 发送标志,通常为 0
  • dest_addr: 目标地址的 sockaddr 结构体指针,用于指定数据发送的目标地址。
  • addrlen: 目标地址结构体的大小。
ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,struct sockaddr *src_addr, socklen_t *addrlen);
  • sockfd: 套接字描述符。
  • buf: 接收数据的缓冲区。
  • len: 缓冲区长度。
  • flags: 接收标志,通常为 0
  • src_addr: 用于存储发送方地址信息的 sockaddr 结构体指针。
  • addrlen: 指向存储发送方地址长度的变量的指针。

close(): 关闭套接字。

int close(int sockfd);

 sockfd: 要关闭的套接字描述符

5.2sockaddr结构

socket API 是一层抽象的网络编程接口 , 适用于各种底层网络协议 , IPv4 IPv6, 以及后面要讲的 UNIX Domain Socket. 然而 , 各种网络协议的地址格式并不相同,

套接字不仅支持跨网络的进程间通信(网络套接字),还支持本地的进程间通信(域间套接字)。在进行跨网络通信时我们需要传递的端口号和IP地址,而本地通信则不需要。网络的设计者想要把跨网络通信和本地通信进行大一统,因此套接字提供了sockaddr_in结构体和sockaddr_un结构体,其中sockaddr_in结构体是用于跨网络通信的(网络套接字),而sockaddr_un结构体是用于本地通信的(域间套接字)。
为了让套接字的网络通信和本地通信能够使用同一套函数接口,于是就出现了sockeaddr结构体,该结构体与sockaddr_in和sockaddr_un的结构都不相同,但这三个结构体头部的16个比特位都是一样的,这个字段叫做协议家族。

​ 

最上面那个是表示是那种套件字(.sin_family)然后是端口号(.sin_port)和IP地址(.sin_addr)

此时当我们在传递在传参时,就不用传入sockeaddr_in或sockeaddr_un这样的结构体,而统一传入sockeaddr这样的结构体。在设置参数时就可以通过设置协议家族这个字段,来表明我们是要进行网络通信还是本地通信,在这些API内部就可以提取sockeaddr结构头部的16位进行识别,如果前16为地址类型是AD_INET,就是网络间通信,如果地址类型是AD_UNIX,就是本地间通信。如上我们就通过通用sockaddr结构,将套接字网络通信和本地通信的参数类型进行了统一。

注意 

  • IPv4和IPv6的地址格式定义在netinet/in.h中,IPv4地址用sockaddr_in结构体表示,包括16位地址类型, 16 位端口号和32位IP地址.
  • IPv4、IPv6地址类型分别定义为常数AF_INET、AF_INET6. 这样,只要取得某种sockaddr结构体的首地址, 不需要知道具体是哪种类型的sockaddr结构体,就可以根据地址类型字段确定结构体中的内容.
  • socket API可以都用struct sockaddr *类型表示, 在使用的时候需要强制转化成sockaddr_in; 这样的好处是程序的通用性, 可以接收IPv4, IPv6, 以及UNIX Domain Socket各种类型的sockaddr结构体指针做为 参数; 

我们可以包含如下四个头文件查看或使用sockaddr、sockaddr_in、in_addr的相关信息:

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>

 sockaddr 结构

sockaddr_in 结构
虽然 socket api 的接口是 sockaddr, 但是我们真正在基于 IPv4 编程时 , 使用的数据结构是 sockaddr_in; 这个结构里主要有三部分信息: 地址类型 , 端口号 , IP 地址

 in_addr结构

 in_addr用来表示一个IPv4IP地址. 其实就是一个32位的整数。

在这个里面我们并没有看到sin_family这个部分,事实上这个就是第二个图片240那行那个在 sockaddr_in 结构体中,sin_family 是 __SOCKADDR_COMMON(sin_) 的一部分。这个设计是为了确保不同的套接字地址结构(例如,IPv4、IPv6等)在内部布局上是一致的,以便于通用的套接字地址处理。

##可以把位于它两边的符号合成一个符号。它允许宏定义从分离的文本片段创建标识符

注: 这样的连接必须产生一个合法的标识符。否则其结果就是未定义的。 所以上面直接被替换成sin_family  

 六、网络字节序

 我们都知道不同的计算机在内存存储中存在大小端问题

  • 大端模式: 数据的高字节内容保存在内存的低地址处,数据的低字节内容保存在内存的高地址处。
  • 小端模式: 数据的高字节内容保存在内存的高地址处,数据的低字节内容保存在内存的低地址处。

磁盘文件中的多字节数据相对于文件中的偏 移地址也有大端小端之分, 网络数据流同样有大端小端之分. 那么如何定义网络数据流的地址呢?

  • 发送主机通常将发送缓冲区中的数据按内存地址从低到高的顺序发出;
  • 接收主机把从网络上接到的字节依次保存在接收缓冲区中,也是按内存地址从低到高的顺序保存;
  • 因此,网络数据流的地址应这样规定:先发出的数据是低地址,后发出的数据是高地址.
  • TCP/IP协议规定,网络数据流应采用大端字节序,即低地址高字节.
  • 不管这台主机是大端机还是小端机, 都会按照这个TCP/IP规定的网络字节序来发送/接收数据;
  • 如果当前发送主机是小端, 就需要先将数据转成大端; 否则就忽略, 直接发送即可;

 

为使网络程序具有可移植性 , 使同样的 C 代码在大端和小端计算机上编译后都能正常运行 , 可以调用以下库函数做网络字节序和主机字节序的转换。

 

  • 这些函数名很好记,h表示host,n表示network,l表示32位长整数,s表示16位短整数。
  • 例如htonl表示将32位的长整数从主机字节序转换为网络字节序,例如将IP地址转换后准备发送。
  • 如果主机是小端字节序,这些函数将参数做相应的大小端转换然后返回 ;
  • 如果主机是大端字节序,这些 函数不做转换,将参数原封不动地返回

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

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

相关文章

计算机网络基本知识(二)

文章目录 概要分层为什么分层怎么分层&#xff1f;1.实体2.协议3.服务 分层基本原则正式认识分层详细例子解释 总结 概要 分层知识&#xff1a;概念理解 分层 为什么分层 大致以上五点 为了解决上面的问题&#xff08;复杂&#xff09; 大问题划分为小问题 怎么分层&#…

新手小白做steam搬砖项目,这些内幕要了解

转眼2024年已经过去了五分之一&#xff0c;很多粉丝都在问steam搬砖项目真的假的&#xff0c;害怕项目的风险&#xff0c;担心steam搬砖项目到底能不能做&#xff0c;所以一直在犹豫和徘徊。我发现很多人想赚钱&#xff0c;但苦于找不到好的副业&#xff0c;高门槛的项目又做不…

C语言的循环结构

目录 前言 1.三种循环语句 1.while循环 2.for循环 2.1缺少表达式的情况 3.do while循环 2.break语句和continue语句 2.1在while循环中 2.2在for循环中 2.3在do while 循环中 3.循环的嵌套 4.go to语句 前言 C语⾔是结构化的程序设计语⾔&#xff0c;这⾥的结构指的是…

【机器学习】数据清洗之识别缺失点

&#x1f388;个人主页&#xff1a;甜美的江 &#x1f389;欢迎 &#x1f44d;点赞✍评论⭐收藏 &#x1f917;收录专栏&#xff1a;机器学习 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共同学习、交流进步…

Verilog刷题笔记23

题目: Suppose you’re building a circuit to process scancodes from a PS/2 keyboard for a game. Given the last two bytes of scancodes received, you need to indicate whether one of the arrow keys on the keyboard have been pressed. This involves a fairly simp…

【LeetCode】51. N 皇后(困难)——代码随想录算法训练营Day30

题目链接&#xff1a;51. N 皇后 题目描述 按照国际象棋的规则&#xff0c;皇后可以攻击与之处在同一行或同一列或同一斜线上的棋子。 n 皇后问题 研究的是如何将 n 个皇后放置在 nn 的棋盘上&#xff0c;并且使皇后彼此之间不能相互攻击。 给你一个整数 n &#xff0c;返回…

day34 本地存储(重要)

目录 本地存储简介本地存储分类—— localStorage本地存储分类—— sessionStorage&#xff08;了解&#xff09;存储复杂数据类型问题1&#xff1a;本地只能存储字符串,无法存储复杂数据类型。问题2&#xff1a;因为本地存储里面取出来的是字符串&#xff0c;不是对象&#xf…

react将选中文本自动滑动到容器可视区域内

// 自动滚动到可视区域内useEffect(() > {const target ref;const wrapper wrapperRef?.current;if (target && wrapperRef) {const rect target.getBoundingClientRect();const wrapperRect wrapper.getBoundingClientRect();const isVisible rect.bottom &l…

消息队列MQ 介绍

&#x1f47d;System.out.println(“&#x1f44b;&#x1f3fc;嗨&#xff0c;大家好&#xff0c;我是代码不会敲的小符&#xff0c;双非大四&#xff0c;Java实习中…”); &#x1f4da;System.out.println(“&#x1f388;如果文章中有错误的地方&#xff0c;恳请大家指正&a…

掌握虚拟化与网络配置之道:深入浅出VMware及远程管理技巧

目录 虚拟机介绍 虚拟机的关键字 服务器架构的发展 为什么用虚拟机VMware 虚拟机和阿里云的区别 功能角度 价格因素 应用场景 优势方面 找到windows的服务管理 配置VMware 关于VMware安装的几个服务 vmware如何修改各种网络配置 关于NAT的详细信息(了解) NAT(网…

我的docker随笔43:问答平台answer部署

本文介绍开源问答社区平台Answer的容器化部署。 起因 笔者一直想搭建一个类似stack overflower这样的平台&#xff0c;自使用了Typora&#xff0c;就正式全面用MarkdownTyporagit来积累自己的个人知识库&#xff0c;但没有做到web化&#xff0c;现在也还在探索更好的方法。 无…

常见的ANSI转义码

ANSI 转义码是一组控制码&#xff0c;用于在文本中添加格式化和颜色。这些码以 ESC&#xff08;Escape&#xff09;字符为开头&#xff0c;通常是 \x1b&#xff0c;后面紧跟着一系列参数和指令。在 ANSI 标准中&#xff0c;这些码通常用于控制终端的文本输出。 下面是一些常见…

springboo冬奥会科普平台源码和论文

随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理平台应运而生&#xff0c;各行各业相继进入信息管理时代&#xf…

cocos creator 3.x 预制体无法显示

双击预制体&#xff0c;进入详情页&#xff0c;没有显示资源 Bomb 是个预制体&#xff0c;但是当我双击进来什么都没有了&#xff0c;无法对预制体进行可视化编辑 目前我只试出来一个解决方法&#xff1a; 把预制体拖进Canvas文件中&#xff0c;这样就能展示到屏幕上&#xff…

为什么要设置止损

2024年1月至2月7日&#xff0c;A股最令人瞩目的事件就是代表小微盘的中证500和中证1000雪球连续敲入&#xff0c;以及万得微盘指数的崩塌&#xff08;1个月下跌50%&#xff09;。 这次的这个过程中&#xff0c;止损很重要。一般情况下&#xff0c;如果设置了20%回撤止损的话&am…

挑战杯 python opencv 深度学习 指纹识别算法实现

1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; python opencv 深度学习 指纹识别算法实现 &#x1f947;学长这里给一个题目综合评分(每项满分5分) 难度系数&#xff1a;3分工作量&#xff1a;4分创新点&#xff1a;4分 该项目较为新颖…

多线程JUC:解决线程安全问题——synchronized同步代码块、Lock锁

&#x1f468;‍&#x1f393;作者简介&#xff1a;一位大四、研0学生&#xff0c;正在努力准备大四暑假的实习 &#x1f30c;上期文章&#xff1a;多线程&JUC&#xff1a;线程的生命周期与安全问题 &#x1f4da;订阅专栏&#xff1a;多线程&JUC 希望文章对你们有所帮…

【DC渗透系列】DC-4靶场

主机发现 arp-scan -l┌──(root㉿kali)-[~] └─# arp-scan -l Interface: eth0, type: EN10MB, MAC: 00:0c:29:6b:ed:27, IPv4: 192.168.100.251 Starting arp-scan 1.10.0 with 256 hosts (https://github.com/royhills/arp-scan) 192.168.100.1 00:50:56:c0:00:08 …

记录 | python list extend()

extend() 函数用于在列表末尾一次性追加另一个序列中的多个值&#xff08;用新列表扩展原来的列表&#xff09;。 以下实例展示了 extend()函数的使用方法&#xff1a; #!/usr/bin/pythonaList [123, xyz, zara, abc, 123]; bList [2009, manni]; aList.extend(bList)print …

大厂聚合支付系统架构演进(下)

点击下方“JavaEdge”&#xff0c;选择“设为星标” 第一时间关注技术干货&#xff01; 关注我&#xff0c;紧跟本系列专栏文章&#xff0c;咱们下篇再续&#xff01; 作者简介&#xff1a;魔都国企技术专家兼架构&#xff0c;多家大厂后端一线研发经验&#xff0c;各大技术社区…