Linux之套接字UDP实现网络通信

news/2024/11/19 20:34:29/

Linux之套接字UDP实现网络通信

文章目录

  • Linux之套接字UDP实现网络通信
  • 1.引言
  • 2.具体实现
    • 2.1需要知道的套接字接口
      • 1.socket()
      • 2.bind()
      • 3.recvfrom()
      • 4.sendto()
    • 2.2服务器端server.hpp
    • 2.3服务器端server.cc
    • 2.4客户端Client.cc

1.引言

​ 套接字(Socket)是计算机网络中实现网络通信的一种编程接口。它提供了应用程序与网络通信之间的一座桥梁,因为它允许应用程序通过网络发送和接收相应的数据以实现不同主机之间的通信。

在这里插入图片描述

通常套接字由以下两部分组成:

1.网络IP和端口号:IP用来标识主机,而端口号可以标识到单台主机的唯一进程。

2.通信协议:套接字通过规定通信协议来制定数据传输和发送的规则。常见的有TCP和UDP等协议。

TCP是一种面向连接的协议,提供可靠的、有序的、基于字节流的数据传输。

UDP是一种无连接的协议,提供不可靠的、无序的、基于数据报的数据传输。

​ 我们今天要实现的是通过UDP协议实现网络通信。UDP协议通信虽然无连接不可靠,可是足够简单到我们了解通信的基本原理。

那么话不多讲,我们赶快看看我们学习完今天这一篇能够实现出怎么样的结果吧:

在这里插入图片描述

我们通过实现客户端和服务器端,实现了通过套接字UDP创建了一个服务器,之后通过客户端链接并且通信的一个功能。

事不宜迟,我们马上实现!

2.具体实现

​ 首先我们需要明确具体的大思路: 先服务器端创建socket套接字,并recvfrom接收到。客户端也创建套接字绑定后确定到唯一IP和端口号之后即可进行通信。

​ 在具体实现之前我们首先需要一些必要的套接字接口

2.1需要知道的套接字接口

1.socket()

​ socket函数是用于创建套接字的函数,创建成功返回文件描述符fd,失败返回-1

int socket(int domain, int type, int protocol);

​ 参数说明:

  • domain

    :指定套接字的地址族(Address Family)

    今天我们选择:

    • AF_INET:IPv4 地址族
  • type

    :指定套接字的类型(Socket Type)

    今天我们选择:

    • SOCK_DGRAM:无连接的数据报套接字,用于 UDP 协议
  • protocol

    :可选参数,指定具体的传输协议。常用的有:

    ​ 今天我们选择:

    • 0:自动选择合适的协议

2.bind()

​ 在Linux下,bind() 函数用于将一个套接字(socket)与特定的IP地址和端口号进行绑定

*int bind(int sockfd, const struct sockaddr addr,socklen_t addrlen);

参数说明:

  • sockfd:要进行绑定的套接字的文件描述符。
  • addr:指向一个 struct sockaddr 结构体的指针,其中包含要绑定的IP地址和端口号信息。
  • addrlenaddr 结构体的长度。

在绑定bind的第二个参数中,我们也需要用到库中定义好的sockaddr_in结构体来初始化!

具体结构体struct sockaddr_in说明:

结构体中有三个值也需要初始化指定一下:

  1. sin_family:表示地址族(Address Family),一般为 AF_INET

  2. sin_port:表示端口号。它是一个 16 位的整数,使用网络字节序(大端字节序)表示。在使用时,通常需要使用 htons() 函数将主机字节序转换为网络字节序。

  3. sin_addr:表示 IPv4 地址。它是一个 struct in_addr 类型的结构体,用于存储 32 位的 IPv4 地址。

    一般服务端用INADDR_ANY,让udp_server在启动时候可以绑定任何ip.

    ​ 客户端用inet_addr函数将字符串转化成32位无符号整数

3.recvfrom()

从套接字接收数据,并获取发送方的地址

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:用于存储发送方的地址信息(对于面向数据报的套接字)。它是一个 struct sockaddr 结构体的指针。
  • addrlensrc_addr 结构体的长度,作为输入参数指定 src_addr 缓冲区的大小,作为输出参数返回实际地址的长度。

4.sendto()

通过套接字发送数据到指定目的地

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:指向目标地址(接收方地址)的结构体指针,可以是 struct sockaddr 或其派生类型的指针。
  • addrlendest_addr 结构体的长度。

2.2服务器端server.hpp

​ 在服务器的头文件中,我们首先需要定义一个udpserver的类,服务器类中需要有服务器的初始化与启动命令,当然需要有构造析构等。默认的私有成员是**_sock套接字和port端口**


const static uint16_t default_port = 8080;class UdpServer
{public:UdpServer(uint16_t port = default_port):_port(port){std::cout<< "server addr: "<<_port <<std::endl;}~UdpServer() {}void InitServer() //初始化服务器{_sock = socket(AF_INET,SOCK_DGRAM,0);if(_sock < 0){std::cerr << " socket create err " << std::endl;}std::cout << "create socket success: " << _sock << std::endl;struct sockaddr_in local; //利用库中创建好的结构体来初始化socketmemset(&local,0,sizeof(local));local.sin_family = AF_INET;local.sin_port = htons(_port); // 本地主机序列转网络序列local.sin_addr.s_addr = INADDR_ANY; //让udp_server在启动时候可以绑定任何ip//绑定if(bind(_sock,(struct sockaddr*)&local,sizeof(local)) < 0){std::cerr << " bind error" << std::endl;exit(-1);}std::cout << "bind socket success: " << _sock << std::endl;}   void Start()    //执行逻辑{char buffer[1024];while(true){   //接收数据//ssize_t recvfrom(套接字,缓冲区,缓冲区大小,flag = 0,client的IP和port,实际结构体大小);struct sockaddr_in far; //远端定义结构体socklen_t len = sizeof(far); int n = recvfrom(_sock,buffer,sizeof(buffer)-1,0,(struct sockaddr*)&far,&len);if(n > 0) buffer[n] = '\0';else continue;std::string clientip = inet_ntoa(far.sin_addr); //ipv4的地址从二进制转化为点分十进制的函数uint16_t clientport = ntohs(far.sin_port);      //将网络字节序转化为一个本地主机字节序std::cout<< clientip << "-" << clientport << "#" << buffer << std::endl;//发送数据//ssize_t sendto(套接字,发的数据,数据大小,flag = 0,(struct sockaddr*)&far,sizeof(far));sendto(_sock,buffer,sizeof(buffer),0,(struct sockaddr*)&far,sizeof(far));}}private:int _sock; //套接字uint16_t _port; //端口};

2.3服务器端server.cc

在服务器端的使用中,我们采用智能指针unique_ptr来帮助资源创建以及销毁,在使用中,我们调用以上server.hpp中类的初始化与启动函数即可.

//输出格式说明:./udp_server portstatic void usage(string proc)//使用手册
{std::cout << "Usage:\n\t" << proc << "port\n" <<std::endl;}int main(int argc,char* argv[]) //获取到参数{if(argc != 2) //若输入参数不是两个的话,就弹出使用手册 {usage(argv[0]);exit(-1);}uint16_t port = atoi(argv[1]); //获取到端口直接进行构造后面std::unique_ptr<UdpServer> ptr(new UdpServer(port));ptr->InitServer();ptr->Start();return 0;
}

2.4客户端Client.cc

在客户端中我们首先需要知道主函数的服务端的ip和端口,也就是我们需要从输入的参数来知道服务端是谁?之后由用户输入消息后发送给服务器端并输出。


// 执行格式:./udp_client ip serverport
static void usage(std::string proc) //使用手册{std::cout << "Usage:\n\t" << proc << "port\n" <<std::endl;}
int main(int argc,char* argv[])
{if(argc != 3)		//如果输入参数个数不是3个就弹出使用手册{usage(argv[0]);exit(-1);}//从主函数获取到了服务端的ip和端口std::string serverip = argv[1];uint16_t serverport = atoi(argv[2]);int sock = socket(AF_INET,SOCK_DGRAM,0);  //创建套接字if(sock < 0){std::cerr << "create socket errno" <<std::endl;exit(-1);}//明确server是谁struct sockaddr_in server;memset(&server,0,sizeof(server));server.sin_family = AF_INET;server.sin_port = htons(serverport);server.sin_addr.s_addr = inet_addr(serverip.c_str());//这里client一定需要绑定bind 不过由os来帮我们做,因为OS需要随机分配端口,防止冲突//用户输入while(true){std::string message;std::cout<< "please Enter#  ";std::cin >> message;sendto(sock,message.c_str(),message.size(),0,(struct sockaddr*)&server,sizeof(server));//接收消息char buffer[1024];struct sockaddr_in temp;socklen_t len = sizeof(temp); int n = recvfrom(sock,buffer,sizeof(buffer)-1,0,(struct sockaddr*)&temp,&len);if(n > 0){buffer[n] = 0;std::cout << "server echo#  " << buffer << std::endl; }}return 0;
}

最后执行后我们便可以看出结果: 说明执行成功!

在这里插入图片描述


http://www.ppmy.cn/news/1047121.html

相关文章

三重奏的和谐:如何完美对齐公司、部门与个人目标

引言 在企业的运营和管理中&#xff0c;目标的设定与对齐是至关重要的。它不仅决定了公司的方向和愿景&#xff0c;还影响到每一个部门和团队成员的工作内容和效果。如何确保公司目标、部门目标和团队个人目标之间的完美对齐&#xff0c;是每一个管理者都需要面对的挑战。 目…

修改 el-select 背景图 样式

1. 原图------------效果图 2. css /***********大的背景图***************/ .el-popper.is-pure {background: url(/src/assets/imgList/memuBG.png) no-repeat;border: none;background-size: 100% 100%; }/*********选中行的字体***********/ .el-select-dropdown__item.s…

如何在Linux系统上搭建自己的FRP内网穿透

前言 我有一个1核1G的服务器有公网IP但是这个1核1G的服务器太垃圾了,几乎什么都跑不起来,不过网速还行,那我本地还有一个物理主机是一个4核4G的,那我就可以把这台主机安装上linux系统当成一个服务器来使用,然后把网络代理到公网IP上.使用内网穿透这篇文章也就出现了. FRP简介 F…

第一讲:BeanFactory和ApplicationContext接口

BeanFactory和ApplicationContext接口 1. 什么是BeanFactory?2. BeanFactory能做什么&#xff1f;3.ApplicationContext对比BeanFactory的额外功能?3.1 MessageSource3.2 ResourcePatternResolver3.3 EnvironmentCapable3.4 ApplicationEventPublisher 4.总结 1. 什么是BeanF…

什么是cURL?

cURL无处不在。它几乎隐藏在所有设备中&#xff0c;例如汽车&#xff0c;蓝光播放器等。它通过互联网协议传输任意类型数据。 在本文中&#xff0c;我们将揭开cURL神秘命令行工具的面纱&#xff0c;解释它是如何成为一种通用代码的&#xff0c;并举例说明其用法。 cURL是什么意…

article-六轴机械臂(带抓手)运动学分析+轨迹规划

1正运动学分析 采用标准的D-h法进行机械腿模型分析&#xff1a; D-h表如下 &#xff08;2&#xff09;通过&#xff08;1&#xff09;求解出机器人各位姿变换矩阵后&#xff0c;求解机器人手臂变换矩阵 ** ** 。通过matlab 计算&#xff0c;写出机器人末端位置。 正运动学分…

安全技术和防火墙

这里写目录标题 安全技术和防火墙一.安全技术1.相关系统2.防火墙的分类 二.防火墙的基本知识1.Netfilter2.防火墙工具介绍2.1 iptables2.2 firewalld2.3 nftables 3.netfilter中五个勾子函数和报文流向 三.iptables1.iptables的组成概述2.相关操作3.添加新的防火墙规则4.查看规…

HCIP生成树STP总结

STP生成树 网桥的4个选举 根网桥&#xff1a; 有且仅有一台&#xff0c;且由BPDU中的桥ID来决定 桥ID 网桥优先级&#xff08;0-65535公有&#xff09; 默认32768 MAC地址&#xff08;只有…