UDP Socket

UDP Socket

网络编程基础概念理解

IP地址: IP地址是互联网上每个设备的唯一标识。

格式:

  • IPv4:4组数字,如 192.168.1.1(共约43亿个,已不足)。
  • IPv6:8组16进制数,如 2001:0db8:85a3::8a2e:0370:7334(解决IPv4不足的问题)。

分类:

  • 公有IP:全球唯一,用于互联网通信。
  • 私有IP:局域网内使用(如 192.168.x.x,10.x.x.x)。

端口号(Port): 区分同一设备上的不同服务或应用程序,标识该主机中的进程。

端口号分类:

  • 知名端口(0-1023):预留给系统服务,如 80(HTTP)、443(HTTPS)、22(SSH)。
  • 注册端口(1024-49151):分配给常见应用,如 3306(MySQL)、3389(远程桌面)。
  • 动态端口(49152-65535):临时分配给客户端程序(如浏览器访问网站时随机生成)。

IP : Port = 标识互联网中该IP对应的主机上对应端口号的进程,也就是套接字Socket。

网络字节序:

计算机在存储多字节数据(如整数、IP地址)时,有两种不同的排列方式:大端序 和 小端序。

网络字节序统一规定网络传输时数据的字节序,为大端序。

为了保证字节排序方式一定相同,从主机发送数据到网络上 和 从网络上接收数据到主机 都需要对数据字节排序方式进行转换。

端口号转换函数:

c 复制代码
#include <arpa/inet.h>  // Linux
// #include <winsock2.h>  // Windows

uint16_t port = 8080;
uint16_t network_port = htons(port);  // 主机序 → 网络序
uint16_t host_port = ntohs(network_port);  // 网络序 → 主机序

IP地址转换函数:

c 复制代码
#include <arpa/inet.h>

// 字符串IP → 网络字节序二进制
const char *ip_str = "192.168.1.1";
struct in_addr ip_bin;
if (inet_pton(AF_INET, ip_str, &ip_bin) != 1) {
    // 转换失败处理
}

// 网络字节序二进制 → 字符串IP
char ip_str_buffer[INET_ADDRSTRLEN];
const char *result = inet_ntop(AF_INET, &ip_bin, ip_str_buffer, sizeof(ip_str_buffer));
if (result == NULL) {
    // 转换失败处理
}

sockaddr 结构,sockaddr_in 结构和sockaddr_un 结构



UDP协议

UDP(User Datagram Protocol 用户数据报协议): 传输层协议,面向数据报,用于需要"高速传输"的场景,例如视频通话、在线游戏等。

核心特性:

  1. 无连接:无需建立连接,直接发送数据(没有三次握手)。
  2. 不可靠传输:不保证数据是否到达、是否按序到达、是否重复。
  3. 轻量高效:没有确认应答、重传、流量控制等机制,传输开销极小,速度极快。
  4. 支持广播/组播:可以一次性向多个设备发送数据(如直播流媒体)。

UDP Socket编程常用接口

创建Socket

接口:socket() 创建通信端点。

c 复制代码
#include <sys/socket.h>

int socket(int domain, int type, int protocol);
c 复制代码
 int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
 
 if (sockfd < 0) {
        perror("socket creation failed");
        exit(EXIT_FAILURE);
    }

绑定IP和端口(仅服务端需要)

接口:bind() 用于将套接字与特定的 IP 地址 和 端口号 绑定,使其能够在该地址和端口上监听或接收数据。

c 复制代码
#include <sys/socket.h>

int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
c 复制代码
 struct sockaddr_in servaddr;
    memset(&servaddr, 0, sizeof(servaddr));
    servaddr.sin_family = AF_INET;
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
    servaddr.sin_port = htons(8080);

 if (bind(sockfd, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0) {
        perror("bind failed");
        exit(EXIT_FAILURE);
    }

注: udp客户端不需要显示绑定,系统会自动进行绑定。

服务器端需要一直固定的IP和端口号,而客户端不需要固定的,只需要在每次连接的时候是固定的就行。

收发数据

接口: sendto() 向指定的目标地址发送数据(UDP 是无连接的,每次需明确目标地址)。

c 复制代码
const char* msg = "Hello from UDP server";
struct sockaddr_in destaddr;
destaddr.sin_family = AF_INET;
destaddr.sin_port = htons(8080);
destaddr.sin_addr.s_addr = inet_addr("192.168.1.100");  // 目标IP

ssize_t n = sendto(sockfd, msg, strlen(msg), 0,
                 (struct sockaddr*)&destaddr, sizeof(destaddr));
                 
if (n < 0) {
    perror("sendto failed");
}

//client发数据给server,虽然客户端不需要绑定,但要发给server还是需要
//在struct sockaddr_in server;之后把服务端的端口号和IP赋值给server
//因为客户端需要知道服务端的 IP:Port

接口: recvfrom() 接收数据并获取发送方的地址信息(UDP 需要获取来源地址以回复)。

c 复制代码
char buffer[1024];
struct sockaddr_in cliaddr;
socklen_t len = sizeof(cliaddr);

ssize_t n = recvfrom(sockfd, buffer, sizeof(buffer), 0,
                    (struct sockaddr*)&cliaddr, &len);
                    
if (n < 0) {
    perror("recvfrom failed");
}

//服务端接收到客户端传来的信息后,cliaddr中就存储进了客户端的IP:端口号

注: recvfrom() 函数会将接收到的数据包中的源地址信息自动填入传进函数的那个结构体中。

关闭连接

接口: close() 用于释放套接字资源。

c 复制代码
close(sockfd);  // Linux
相关推荐
网硕互联的小客服3 小时前
如何利用Elastic Stack(ELK)进行安全日志分析
linux·服务器·网络·安全
浩浩测试一下4 小时前
Authpf(OpenBSD)认证防火墙到ssh连接到SSH端口转发技术栈 与渗透网络安全的关联 (RED Team Technique )
网络·网络协议·tcp/ip·安全·网络安全·php
leagsoft_10035 小时前
联软NSPM自动化策略管理 助力上交所加速国产化替代提升运维效率
运维·网络·自动化
Think Spatial 空间思维5 小时前
【实施指南】Android客户端HTTPS双向认证实施指南
android·网络协议·https·ssl
昔我往昔6 小时前
https和http有什么区别-http各个版本有什么区别
网络协议·http·https
漫步者TZ6 小时前
【Netty系列】解决TCP粘包和拆包:LengthFieldBasedFrameDecoder
java·网络协议·tcp/ip·netty
leagsoft_10036 小时前
筑牢企业网管域安全防线,守护数字核心——联软网管域安全建设解决方案
网络·安全·网络安全
苦学编程的谢7 小时前
Java网络编程API 1
java·开发语言·网络
alien爱吃蛋挞7 小时前
【JavaEE】万字详解HTTP协议
网络·网络协议·http
Bright16689 小时前
mkcert实现本地https
网络协议·http·https