Linux TCP 和 UDP 通信

TCP与UDP

TCP与UDP对比

TCP

特点:面向连接、可靠传输、流量控制、拥塞控制、全双工

优点:可靠性(ACK确认、超时重传)、顺序性

缺点:首部开销大(20字节)、连接管理复杂、延迟高

应用场景:HTTP/HTTPS、文件传输(FTP)、邮件(SMTP/IMAP)

UDP

特点:无连接、尽力交付、无重传、无拥塞控制、低开销

优点:传输效率高(8字节首部)、实时性强

缺点:数据可能丢失、乱序

应用场景:视频流(RTP)、DNS查询、在线游戏、VoIP

TCP通信

TCP网络编程流程

服务器流程:

1.创建流式套接字--socket()

2.填充服务器的网络信息结构体--struct sockaddr_in

3.将套接字于服务器的网络信息结构体绑定--bind()

4.将套接字设置成被动监听状态--listen()

5.阻塞等待客户端连接--accept()

6.收发数据---recv()/send()

7.关闭套接字--close()

socket编程

socket函数

头文件:#include <sys/socket.h>

函数原型:int socket(int domain, int type, int protocol);

参数:

•domain就是通信域,也可以说是协议族,常见通信域如下:

•AF_INETIPV4

•AF_INET6IPV6

•AF_PACKET原始套接字使用

•AF_UNIX本地通信

•AF_LOCAL本地通信

•type表示socket的类型,常用的socekt类型如下:

•SOCK_STREAMTCP使用

•SOCK_DGRAMUDP使用

•SOCK_RAW原始套接字

•protocol表示附加协议无附加协议填0(自动选择默认协议)

返回值:成功返回套接字(文件描述符-非负整数),失败返回-1(并重置错误码)

bind函数

头文件:#include <sys/socket.h>

函数原型:int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);

参数:

•sockfd就是文件描述符(由socket函数返回)

•addr表示地址结构体指针(const struct sockaddr*),指向要绑定给sockfd的协议地址,该结构体根据创建socket时的协议族不同而不同,这里以IPV4为例:

cs 复制代码
struct sockaddr_in {
    sa_family_t sin_family;   // 地址族
    in_port_t sin_port;       // 16位端口号
    struct in_addr sin_addr;  // 32位IP地址
};

struct in_addr {
    uint32_t s_addr;          // 32位IPv4地址
};

•addrlen表示地址的长度

返回值:成功返回0,失败返回-1(并重置错误码)

listen函数

头文件:#include <sys/socket.h>

函数原型:int listen(int sockfd, int backlog);

参数:

•sockfd表示要监听的socket套接字

•backlog表示半连接队列的长度(即socket可以排队的最大连接个数)

返回值:成功返回0,失败返回-1(并重置错误码)

accept函数

头文件:#include <sys/socket.h>

函数原型:int accept(int sockfd, struct sockaddr*addr, socklen_t*addrlen);

参数:

•sockfd表示处于监听状态的socket

•addr用于保存客户端地址的结构体指针,如果不关心客户端的信息,可以传NULL

•addrlen输入时为addr的缓冲区大小,输出时为实际地址长度,如果不关心客户端的信息,可以传NULL

返回值:成功返回新的socket文件描述符(由内核生成,代表着与返回客户端的TCP连接,专用于与客户端通信),失败返回-1(并重置错误码)

connect函数

头文件:#include <sys/socket.h>

函数原型:int connect(int sockfd, const struct sockaddr*addr, socklen_taddrlen);

参数:

•sockfd表示客户端的socket套接字

•addr表示目标服务器的地址结构体

•addrlen地址结构体的长度

返回值:成功返回0,失败返回-1(并重置错误码)

recv函数

头文件:#include <sys/socket.h>

函数原型:ssize_t recv(int sockfd, void *buf, size_tlen, int flags);

参数:

•sockfd表示客户端的socket套接字

•buf用于存储接收的数据

•len数据大小

•flags控制选项(如MSG_DONTWAIT非阻塞,MSG_PEEK窥视数据,0阻塞)

返回值:成功返回实际接收数据字节数,失败返回-1(并重置错误码)

send函数

头文件:#include <sys/socket.h>

函数原型:ssize_tsend(int sockfd, const void *buf, size_tlen, int flags);

参数:

•sockfd表示客户端的socket套接字

•buf要发送的数据首地址

•len数据大小

•flags控制选项(如MSG_DONTWAIT非阻塞,MSG_PEEK窥视数据,0阻塞)

返回值:成功返回实际发送数据字节数,失败返回-1(并重置错误码)

close函数

头文件:#include <unistd.h>

函数原型:int close(int fd);

参数:

•fd表示要关闭的套接字

返回值:成功返回0,失败返回-1

TCP网络编程流程

客户端流程:

1.创建流式套接字--socket()

2.填充服务器的网络信息结构体--struct sockaddr_in

3.与服务器建立连接--connect()

4.收发数据---recv()/send()

5.关闭套接字--close()

完整TCP连接建立

完整TCP连接建立过程

四次挥手

UDP通信

UDP通信流程

sendto/recvfrom

recvfrom

recvfrom函数

头文件:#include <sys/socket.h> #include <sys/types.h>

函数原型:ssize_t recvfrom(int sockfd, void *buf, size_t len, int flags,struct sockaddr *src_addr, socklen_t *addrlen);

参数:

•sockfd表示客户端的socket套接字

•buf要接收的数据首地址

•len可接受数据的最大长度

•flags控制选项(如MSG_DONTWAIT非阻塞,MSG_PEEK窥视数据,0阻塞)

•src_addr源地址,获取发送方的信息

•addrlen地址长度

返回值:成功返回接收字节数,失败返回-1(并重置错误码)

sendto

sendto函数

头文件:#include <sys/socket.h> #include <sys/types.h>

函数原型:ssize_t sendto(int sockfd, const void *buf, size_t len, int flags,const struct sockaddr *dest_addr, socklen_t addrlen);

参数:

•sockfd表示客户端的socket套接字

•buf要发送的数据首地址

•len数据大小

•flags控制选项(如MSG_DONTWAIT非阻塞,MSG_PEEK窥视数据,0阻塞)

•dest_addr目的地址,数据将要发向哪一个ip地址的主机

•addrlen地址长度

返回值:成功返回发送字节数,失败返回-1(并重置错误码)

单播

单播:一对一通信,数据包从单一源地址发送到单一目标地址

流程与各层职责:

应用层

•调用sendto()发送数据到指定目标IP和端口

•调用recvfrom()接收来自特定源的数据

传输层

•封装UDP头部:源端口、目标端口、长度、校验和

•不建立连接,直接发送数据报

网络层

•封装IP头部:源IP、目标IP(单播地址,如192.168.1.100)

•根据目标IP查找路由表,选择下一跳

数据链路层

•根据目标IP的MAC地址(通过ARP解析)封装以太网帧

•通过物理网络设备(如网卡)发送到目标主机

组播

组播:一对多通信,数据包发送到一个组播组,组内所有成员均可接收

流程与各层职责:

应用层

•发送端:调用发送数据到组播地址(如239.255.0.1)

•接收端:调用加入组播组(IP_ADD_MEMBERSHIP)

传输层

•封装UDP头部:目标端口为组播端口(如12345)

•组播成员无需提前建立连接

网络层

•封装IP头部:目标IP为组播地址(D类地址,224.0.0.0~239.255.255.255)

•接收端通过IGMP报文通知路由器加入/离开组播组

•路由器维护组播组成员列表,仅向存在成员的子网转发数据

数据链路层

•组播MAC地址映射:将IP组播地址转换为以太网组播MAC(如01:00:5E:XX:XX:XX)

•交换机/路由器:根据组播MAC地址复制数据包到多个端口

setsockopt

setsockopt函数

头文件:#include <sys/socket.h>

函数原型:int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen);

参数:

•sockfd目标套接字描述符,表示需要配置的套接字

•level选项层级(如SOL_SOCKET、IPPROTO_IP)

•optname选项名称(如SO_REUSEADDR、IP_ADD_MEMBERSHIP)

•optval指向选项值的指针,类型和长度需与optname匹配

•optlenoptval指向的数据长度

返回值:成功返回0,失败返回非0数据

广播

广播:一对所有通信,数据包发送到同一网络内的所有主机

流程与各层职责:

应用层

•发送端:调用setsockopt()启用广播选项(SO_BROADCAST)

•调用sendto()发送数据到广播地址(如255.255.255.255)

传输层

•封装UDP头部:目标端口为广播端口(如9999)

•接收端无需加入组,但需监听指定端口

网络层

•封装IP头部:目标IP为广播地址(受限广播255.255.255.255或定向广播192.168.1.255)

•受限广播仅在本局域网内传播,路由器默认不转发

•定向广播可跨子网(需路由器支持,通常被禁用)

数据链路层

•广播MAC地址:FF:FF:FF:FF:FF:FF

•交换机将广播包泛洪到所有端口(除源端口)

相关推荐
SkyWalking中文站4 小时前
认识 Horizon UI · 5/17:3D 基础设施地图
运维·监控·自动化运维
tntxia21 小时前
linux curl命令详解_curl详解
linux
扛枪的书生1 天前
Linux 网络管理器用法速查
linux
SkyWalking中文站1 天前
认识 Horizon UI · 1/17:SkyWalking 新一代可观测性控制台
运维·前端·监控
顺风尿一寸1 天前
Java Socket 内核之旅:从 SocketChannel.read() 到 tcp_recvmsg 与 epoll 的完整调用链路
linux
雪梨酱QAQ1 天前
Kubeneters HA Cluster部署
运维
江华森1 天前
Spring Cloud 微服务全栈实战:从 Eureka 到 Docker Compose 一文贯通
运维
江华森1 天前
Matplotlib 数据绘图基础入门
运维
XIAOHEZIcode1 天前
Ubuntu 终端美化全栈指南:Bash 到 Kitty 踩坑实录
linux·ubuntu·命令行
江华森1 天前
NumPy 数值计算基础入门
运维