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

Linux网络编程——UdpServer - 实践

 服务端基本框架

#include 
#include "LOG.hpp"
static const int defaultsockfd = -1;
class UdpServer
{
public:UdpServer(uint16_t port)//构造:_sockfd(defaultsockfd),_port(port),_isrunning(false){}//服务初始化void InitServer(){}//开始运行void Start(){}~UdpServer()//析构{}
private:int _sockfd;uint16_t _port;bool _isrunning;//判断服务端是否运行
};

为什么没有ip呢?到时候IP整合在InitAddr.hpp中

 UdpServer.hpp

#include 
#include 
#include 
#include 
#include 
#include
#include "LOG.hpp"
enum
{SOCKET_ERROR=1,BIND_ERROR,USAGE_ERROR
};
const static int defaultsockfd = -1;
class UdpServer
{
public:UdpServer(uint16_t port)//构造:_sockfd(defaultsockfd),_port(port),_isrunning(false){}//服务初始化void InitServer(){//创建套接字_sockfd = socket(AF_INET,SOCK_DGRAM,0);if(_sockfd<0)//创建失败,日志报错,退出{LOG(FATAL,"socket error:%s,%d\n",strerror(errno),errno);exit(SOCKET_ERROR);}//创建成功LOG(INFO,"socket create success,sockfd:%d\n",_sockfd);//填充sockaddr_in结构struct sockaddr_in local;memset(&local,0,sizeof(local));//清一下结构里面的数据local.sin_family=AF_INET;//网络通信//port需要经过网络传输给对面的client,主机序列要先转网络序列local.sin_port=htons(_port);local.sin_addr.s_addr=INADDR_ANY;//bindint n = bind(_sockfd,(struct sockaddr*)&local,sizeof(local));if(n<0)//绑定失败{LOG(FATAL,"bind error,%s:%d\n",strerror(errno),errno);exit(BIND_ERROR);}//绑定成功LOG(INFO, "socket bind success\n");}//开始运行void Start(){_isrunning = true;while(_isrunning)//服务一般都是一直运行的死循环{LOG(DEBUG,"sever is running...\n");sleep(1);}}~UdpServer()//析构{}
private:int _sockfd;uint16_t _port;bool _isrunning;//判断服务端是否运行
};

填充sockaddr_in记得引头文件<apra/inet.h> ;struct sockaddr_in是系统提供的数据类型

端口号port主机序列转成网络序列用htons()

在网络编程中,当一个进程需要绑定一个网络端口以进行通信时,可以使用 INADDR_ANY 作为 IP 地址参数。这样做意味着该端口可以接受来自任何 IP 地址的连 接请求,无论是本地主机还是远程主机。

那么local还只是对象,还在栈上开辟空间,所以InitServer初始化函数只有创建套接字socket以及填充sockaddr_in是不够的,需要进行绑定bind ,bind完成后server的初始化才算完成。

下面就可以写启动server的函数了;现在的软件若不是用户主动退出,就会一直运行,服务器也一样会一直运行,直到管理者不想运行了。所以Server都是死循环,那么就需要一个对象来判断服务器是否一直再跑(_isrunning)

 Server.cc

#include
#include"UdpServer.hpp"
#include 
void Usage(std::string a)
{std::cout<<"Usage:\n\t"< udps = std::make_unique(port);//初始化udps->InitServer();//启动udps->Start();return 0;
}

测试一下啊 

没有问题

下面我们来完成一些Udp服务的启动

1.我们需要让Server端先收数据

2.我们要让Server将收到的数据再发回去 

接收可以用recvfrom

recvfrom() 函数是一个系统调用,用于从套接字接收数据。该函数通常与无连接的数据报服务(如 UDP)一起使用,但也可以与其他类型的套接字使用。

man recvfrom 

参数

sockfd:一个已打开的套接字的描述符。

buf:一个指针,指向用于存放接收到的数据的缓冲区(输出型参数对应的缓冲区)。

len:缓冲区的大小/期望长度(以字节为单位)。

flags:读取方式。通常可以设置为0,但以下是一些可用的标志:

        MSG_WAITALL:尝试接收全部请求的数据。函数可能会阻塞,直到收到所有数据。
        MSG_PEEK:查看即将接收的数据,但不从套接字缓冲区中删除它【1】。
        其他一些标志还可以影响函数的行为,但在大多数常规应用中很少使用。

有人给你发消息你想不想知道他是谁?为什么?

当然想,我还要给他回消息呢。

那么通过什么信息知道对方是谁呢?socket:IP和Port(通过以下参数就可以知道是谁发的)
src_addr:一个指针,指向一个 sockaddr 结构,用于保存发送数据的源地址。

addrlen:一个值-结果参数。开始时,它应该设置为 src_addr 缓冲区的大小。当 recvfrom() 返回时,该值会被修改为实际地址的长度(以字节为单位)。

返回值 ssize_t

在成功的情况下,recvfrom() 返回接收到的字节数。
如果没有数据可读或套接字已经关闭,那么返回值为0。
出错时,返回 -1,并设置全局变量 errno 以指示错误类型。

服务端接收完如何发送呢?

可以用sendto

sendto() 函数是一个系统调用,用于发送数据到一个指定的地址。它经常与无连接的数据报协议,如UDP,一起使用。不像 send() 函数只能发送数据到一个预先建立连接的远端,sendto() 允许在每次发送操作时指定目的地址

man sendto

sendto 和recvfrom用法差不多

以下参数:

sockfd:一个已打开的套接字的描述符。

buf:一个指针,指向要发送的数据的缓冲区。

len:要发送的数据的大小(以字节为单位)。

flags:控制发送行为的标志。通常可以设置为0。

下面两个参数决定发给谁
dest_addr:指向 sockaddr 结构的指针,该结构包含目标地址和端口信息。

addrlen:dest_addr 缓冲区的大小(以字节为单位)。

返回值:

成功时,sendto() 返回实际发送的字节数。
出错时,返回 -1 并设置全局变量 errno 以指示错误类型。

服务端启动方法实现

void Start(){//之前提到的判断服务是否还在运行的变量_isrunning = true;while(_isrunning)//服务一般都是一直运行的死循环{//接收char buffer[1024];struct sockaddr_in peer;//对client端构建sockaddrsocklen_t len = sizeof(peer);//必须初始化为sizeof(peer)/peer长度ssize_t n = recvfrom(_sockfd,buffer,sizeof(buffer)-1,0,(struct sockaddr*)&peer,&len);if(n > 0)//收到了{buffer[n]=0;//在接收数据的末尾手动添加空字符\0LOG(DEBUG,"服务端获取信息:%s\n",buffer);//你收到了才能发嘛sendto(_sockfd,buffer,strlen(buffer),0,(struct sockaddr*)&peer,len);}}_isrunning = false;}

 客户端和服务端的步骤差不多,只是客户端不显示绑定

client.cc

#include
#include
#include 
#include 
#include 
#include 
#include 
void Usage(std::string a)
{std::cout<<"Usage:\n\t"< 0){buffer[n]=0;std::cout<<"服务端返回#"<

client 要不要进行 bind? 一定要 bind

但是,不需要显示 bind,client 会在首次发送数据的时候会自动进行 bind

为什么?因为server 端的端口号,一定是不可改变的,client 需 要 port,bind 随机端口.

为什么?因为客户端 会非常多. 防止大量客户端bind端口后对大量端口进行占用

client 需要 bind,但是不需要显示 bind,让本地 OS 自动随机 bind, 选择随机端口号

测试代码 

封装一下addr,方便获取Client端的ip和端口

#include 
#include 
#include 
#include 
#include 
class InetAddr
{
private:GetAddr(std::string *ip,uint16_t *port){*port=ntohs(_addr.sin_port);//网络转主机//将32位网络字节序的IPv4地址转换为点分十进制字符串*ip = inet_ntoa(_addr.sin_addr);}
public:InetAddr(struct sockaddr_in addr)//构造:_addr(addr){}std::string Ip(){return _ip;}uint16_t Port(){return _port;}~InetAddr(){}
private:struct sockaddr_in _addr;uint16_t _port;std::string _ip;
};

http://www.rkmt.cn/news/26540.html

相关文章:

  • 2025年最新游戏机和游艺机的屏幕驱动方案(含音乐播放和功放芯片)
  • 2025 年最新推荐!国内加工厂家排行榜:含车铣复合 / 精密零件 / CNC 车床等领域优质企业
  • 【安徽财经大学主办】第七届管理科学信息化与经济创新发展国际学术会议 (MSIEID 2025)
  • 通过HTML演示JVM的垃圾回收-新生代与老年代 - 详解
  • 2025年冷水机组厂家权威推荐榜:水冷螺杆/风冷螺杆/水冷式/风冷式/螺杆式冷水机专业选购指南
  • 再看 AI 网关:助力 AI 应用创新的关键基础设施
  • git pull中有 merge功能解释
  • 2025年工厂维保服务厂家权威推荐榜:机电维修、应急维修、设备安装维修、运维服务全方位解决方案精选
  • 软件工程第三次作业——结对项目
  • 博弈树
  • 2025 年活性炭源头厂家最新推荐榜,技术实力与市场口碑深度解析,筛选优质可靠品牌颗粒/柱状/粉末/煤质/木质活性炭
  • 2025年拖鞋机厂家权威推荐榜:酒店拖鞋生产线,全自动拖鞋机,一次性拖鞋机,酒店一次性拖鞋机器专业选购指南
  • 【2025最新】ArcGIS for JS 实现地图卷帘效果,动态修改参数(进阶版) - 教程
  • A-Beta 剪枝
  • https://vscode-elements.github.io/components/toolbar-button/
  • 【深度学习计算机视觉】10:转置卷积 - 详解
  • 2025年CNC加工厂家权威推荐榜:CNC精密加工/加工中心CNC/cnc电脑锣加工/铝板cnc加工/精密CNC加工源头企业综合评测
  • 基于 tar.gz 的自定义 安装InfluxDB
  • 论微服务架构设计及其应用(AI写作)
  • 2025年润滑油厂家权威推荐榜:工业润滑油,汽车润滑油,发动机润滑油,甲醇发动机润滑油,三特/三球/安迪森全合成润滑油,中国长效润滑油品牌精选
  • 平滑滚动到页面元素scrollIntoView
  • 浏览器检查源代码出现如下问题解决方法
  • 2025年兄弟机床维修厂家权威推荐榜:专业维修技术与高效服务口碑深度解析
  • 2025年TYPE-C母座厂家权威推荐榜:防水/板上/沉板/立插/卧式/侧贴/贴片式/插件式全系列,5A大电流高速TID认证接口一站式供应
  • 配置即权限:从传统开源 RBAC 框架到 SPARK 的六层资料护盾,告别改权限就要改代码的魔咒
  • 深入解析:【数据结构】顺序表0基础知识讲解 + 实战演练
  • 比特币挖矿盈利能力9月下降超7%
  • Nimm Game
  • 基于C++的远程键盘监控器设计与实现 - 教程
  • 2025年医药冷链运输厂家权威推荐榜:药品/临床样本/CAR-T/蛋白/诊断试剂/生物制品/血液/细胞/芯片全程温控,冷藏车/冷藏箱/保温箱/干冰/液氮及国际冷链进出口专业服务