当前位置: 首页 > news >正文

Linux网络:UDP socket创建流程与简单通信

本文介绍 UDP 服务端与客户端 的创建流程,和相关的函数接口

核心流程

  1. 创建 socket → socket()
  2. 填写服务器地址信息 → sockaddr_in 结构体
  3. 绑定地址和端口 → bind()
  4. 接收并响应客户端数据 → recvfrom() / sendto()

在这里插入图片描述

socket()

#include<sys/socket.h>
int socket(int domain, int type, int protocol);
参数说明
domain地址族(协议族),常见值:
AF_INET:IPv4
AF_INET6:IPv6
AF_UNIX:本地通信(进程间通信)
type套接字类型,决定通信方式:
SOCK_STREAM:面向连接(TCP)
SOCK_DGRAM:无连接(UDP)
protocol一般写 0,表示让系统自动选择适合给定 domaintype 的协议

使用示例

int sockfd = socket(AF_INET, SOCK_DGRAM, 0);
//成功返回文件描述符
//失败返回-1
if (sockfd < 0)
{std::cerr << "socket error" << std::endl;
}

sockaddr_in

他有4个成员,赋值前三个即可

struct sockaddr_in {sa_family_t sin_family;   // 地址族,必须是 AF_INETuint16_t sin_port;        // 端口号(网络字节序)struct in_addr sin_addr;  // IP 地址char sin_zero[8];         // 填充字节,保持与 sockaddr 一致
};

在赋值时需要注意,

  1. 端口号要转换为网络序列
  2. IP地址调用inet_addr
    server.sin_port = htons(serverport);                  // 主机序列转网络序列server.sin_addr.s_addr = inet_addr(serverip.c_str()); // 转换4字节

INADDR_ANY

服务器端的服务需要固定的端口
而IP地址给INADDR_ANY,表示监听任意IP地址,即从哪个网卡发来哪个请求都可以处理

其实和手动给0或者0.0.0.0 作用相似

bind()

服务器端和客户端都需要将套接字和本地地址(IP+port)绑定,才能做到接收和转发消息

int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
参数含义
sockfdsocket() 创建的套接字文件描述符
addr本地地址结构体(sockaddr* 类型,实际通常传 sockaddr_in* 转换而来)
addrlen结构体 addr 的大小(用 sizeof(sockaddr_in)

使用示例

int n = bind(_sockfd, (struct sockaddr *)&addr, sizeof(addr));
//成功返回0,失败返回-1
if (n < 0)
{LOG(FATAL, "bind error, %s, %d\n", strerror(errno), errno);exit(BIND_ERROR);
}

客户端的bind()

客户端也是需要绑定的,不然如何发送消息呢。

但是,不需要我们手动调用bind(),
在第一次发送请求的时候,OS自动调用bind()

当你第一次调用:  sendto()操作系统会自动调用 bind() 来:1.分配一个临时的本地 IP(通常是默认网卡的 IP)2.分配一个 可用的随机端口(称为 ephemeral port)

recvfrom()

recvfrom() 是 UDP 套接字编程中用来接收数据报的核心函数,
它不仅接收数据,还能告诉你数据是从哪个客户端发来的

#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一般设置为 0,特殊需求可用 MSG_PEEK(窥视)、MSG_WAITALL
src_addr输出参数,对方的地址结构体(可获取对方 IP 和端口)
addrlen输入输出参数,传入结构体长度,返回时写入实际地址大小

使用示例

struct sockaddr_in peer;
socklen_t len = sizeof(peer);
char buffer[1024];
ssize_t n = recvfrom(_sockfd, buffer, sizeof(buffer) - 1, 0, (struct sockaddr *)&peer, &len);
//成功返回实际接收的字节数(就是多少个英文字符)
//失败返回-1
//peer存客户端的数据

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);

有区别的是len、det_addr

参数含义
sockfd套接字文件描述符(由 socket() 创建)
buf要发送的数据缓冲区的指针
len要发送的数据字节数
flags通常为 0,特殊用途可以设置为 MSG_CONFIRM
dest_addr目标地址结构体(例如 sockaddr_in,需强转为 sockaddr*
addrlendest_addr 的长度(如 sizeof(sockaddr_in)

使用示例

struct sockaddr_in peer;
socklen_t len = sizeof(peer);
//客户端要持续运行,所以给死循环
while (true)
{char buffer[1024];ssize_t n = recvfrom(_sockfd, buffer, sizeof(buffer) - 1, 0, (struct sockaddr *)&peer, &len);if (n > 0){buffer[n] = {0};InetAddr addr(peer);	//自己写的类,为了获取转换后的网络字节序和4字节序LOG(DEBUG, "get message from [%s:%d]:%s\n", addr.Ip(), addr.Port(), buffer);sendto(_sockfd, buffer, strlen(buffer), 0, (struct sockaddr *)&peer, len);}
}

小结

介绍了socket创建流程需要的接口,以及在这方面服务器端和客户端的区别

http://www.dtcms.com/a/269705.html

相关文章:

  • 手机能用酒精擦吗?
  • 前端学习3--position定位(relative+absolute+sticky)
  • Android kotlin 协程的详细使用指南
  • SpringBoot校园外卖服务系统设计与实现源码
  • EXCEL链接模板无法自动链接到PowerBI?试试这个方法
  • 自动驾驶的“安全基石”:NVIDIA如何用技术守护未来出行
  • 最新 HarmonyOS API 20 知识库 重磅推出
  • 【计算机网络】王道考研笔记整理(1)计算机网络体系结构
  • 嘉立创黄山派下载watch ui demo 教程(sf32)
  • Modbus TCP转Profinet网关实现视觉相机与西门子PLC配置实例研究
  • OpenCV 图像哈希类cv::img_hash::AverageHash
  • ​扣子Coze飞书多维表插件通用参数和通用返回值
  • Mysql常用内置函数,复合查询及内外连接
  • 利用外部Postgresql及zookeeper,启动Apache Dolphinscheduler3.1.9
  • 小程序订阅消息设计:用户触达与隐私保护的平衡法则
  • STM32-定时器
  • 冷链物流配送中心选址与路径优化模型研究
  • SpringBoot基础特性:自定义Banner与Swagger文档
  • houdini云渲染和云解算新选择
  • 易境通海运系统:赋能海运企业全链路数字化运营!
  • 【Note】《Kafka: The Definitive Guide》第11章:Stream Processing
  • BERT代码简单笔记
  • C#中封装halcon函数的报错
  • 代码详细注释:C语言实现控制台用户注册登录系统
  • Google AI 刚刚开源 MCP 数据库工具箱,让 AI 代理安全高效地查询数据库
  • 前后端分离(java) 和 Nginx在服务器上的完整部署方案(redis、minio)
  • JxBrowser 7.43.4 版本发布啦!
  • 人工智能驱动下的可再生能源气象预测:构建绿色能源时代的新大脑
  • 微服务化采集平台:可扩展性与容错机制
  • 相机Camera日志实例分析之五:相机Camx【萌拍闪光灯后置拍照】单帧流程日志详解