首页 > 其他分享 >UDP通信 [补档-2023-07-22]

UDP通信 [补档-2023-07-22]

时间:2024-01-13 18:11:07浏览次数:36  
标签:serv UDP 07 22 cfd 接字 include buf

UDP通信

6-1 简介

​ UDP通信是面向无链接的,不稳定,不可靠,不安全的一种通信方式。TCP在通信前发送方会向接收方进行三次握手链接,然后确认双方链接后才会进行数据传输,最后四次挥手保证链接关闭。而UDP不会三次握手四次挥手,它会直接向发送方发送数据,无论接收方是否会收到,所以UDP更适合在稳定的网络环境中使用。

6-2 UDP通信相关函数

6-2-1 recvfrom函数

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

函数原型: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:返回发送方的地址信息,包括ip和端口号

addrlen:表示scr_addr指向的缓冲区大小,作为输出参数表示实际发送方地址的长度。

函数返回值

成功:返回接收的数据的字节数量。

失败:返回-1。

发送错误:返回-1

6-2-2 sendto函数

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

函数原型: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:指向目标地址的指针,其中包含目标ip和端口号信息。

addrlen:目标地址结构体的长度。

函数返回值

成功:返回实际发送的字节数。

错误:返回-1。

6-3 使用UDP通信的服务端编写步骤

​ 第一步:创建一个套接字,但是socket函数的第二个参数要填 SOCK_DGRAM来将协议改为UDP

​ 第二步:将刚才创建的套接字与本地地址绑定。

​ 第三步:进入一个循环然后使用recvfrom函数来接收数据,接收到数据可以处理数据,然后使用sendto将处理好的数据回应给客户端。

​ 第四步:重复第三步,如果服务端需要关闭了记得关闭掉第一步创建的套接字描述符。

6-4 使用UDP通信的客户端编写步骤

​ 第一步:创建一个套接字,但是socket函数的第二个参数要填 SOCK_DGRAM来将协议改为UDP

​ 第二步:可以根据实际开发的需要,进入一个循环体持续收发数据或者是只收发一次数据。收发数据的函数就是上面的那两个。

​ 第三步:重复第二步,如果不需要发送数据了记得关闭第一步创建的套接字。

6-5 服务端代码示例

点击查看代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <netiner/in.h>
#include <ctype.h>
#include <sys/socket.h>

 

int main()

{

  //创建套接字,记得使用SOCK_DGRAM

  int cfd = socket(AF_INET, SOCK_DGRAM,0);

  if (cfd < 0) {

​     perror("创建套接字失败!");

​     return -1;

  }

  

  struct sockaddr_in serv;//用来保存服务端地址信息,用来绑定地址

  struct sockaddr_in clie;//用来保存客户端地址信息,一会用于通信

  serv.sin_family = AF_INET;

  serv.sin_port = htons(10066);

  serv.sin_addr.s_addr = htonl(INDAAR_ANY);

  bind(cfd, (struct sockaddr*)&serv, sizeof(serv));

 

  int n;//用来记录recvfrom从客户端接收了多少个字节的数据

  char data[100];//用来存储来自客户端的信息

  char serv_data[100] = "服务端已经接收到数据"; //服务端的回应

  socklen_t len;//用来记录客户端地址信息的大小

  while (1) {

​     //读数据

​     memset(data, 0x00, sizeof(data));//初始化

​     len = sizeof(clie);//记录大小

​     //接收数据

​     n = recvfrom(cfd, data, szieof(data), 0, (struct sockaddr*)&clie, &len);

 

​     //发数据(处理数据)

​     sendto(cfd, serv_data, n, 0, (struct sockaddr *)&clie,len);

  }

  //关闭套接字

  close(cfd);

}

6-6 客户端代码示例

点击查看代码
//udp客户端
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <netinet/in.h>
#include <ctype.h>

 

int main()

{

  //创建socket

  int cfd = socket(AF_INET, SOCK_DGRAM, 0);

  if (cfd < 0)

  {

​     perror("socket error");

​     return -1;

  }

 

  int n;

  char buf[1024];

  struct sockaddr_in serv;

  serv.sin_family = AF_INET;

  serv.sin_port = htons(8888);

  inet_pton(AF_INET, "127.0.0.1", &serv.sin_addr.s_addr);

 

  while (1)

  {

​     //读标准输入数据

​     memset(buf, 0x00, sizeof(buf));

​     n = read(STDIN_FILENO, buf, sizeof(buf));

 

​     //发送数据

​     sendto(cfd, buf, n, 0, (struct sockaddr*)&serv, sizeof(serv));

 

​     //读取数据

​     memset(buf, 0x00, sizeof(buf));

​     n = recvfrom(cfd, buf, sizeof(buf), 0, NULL, NULL);

​     printf("n==[%d], buf==[%s]\n", n, buf);

  }

 

  //关闭套接字

  close(cfd);

 

  return 0;

}

标签:serv,UDP,07,22,cfd,接字,include,buf
From: https://www.cnblogs.com/xiaobai1523/p/17962701

相关文章

  • socket编程 [补档-2023-07-10]
    Linux网络编程1.socket编程socket是一种通信机制,用于在网络中不同计算机之间进行数据传输,当然也可用用于进程间通信。在linux中,有文件描述符这么个东西,我们可以通过socket函数创建一个网络连接,socket的返回值为一个文件描述符,我们拿到这个文件描述符就可以像操作普通io文件那样......
  • P9007 [入门赛 #9] 最澄澈的空与海 (Hard Version) 题解
    Updon2023.10.1408:21:修改了推式子和题意的一些小错误。前言一道恐怖的绿题。显然我认为应该是蓝题。(不过在这篇题解写到一半的时候升蓝了,感谢@StudyingFather。)名字挺好的。题意给定\(n\),求出满足以下条件的三元组\((x,y,z)\)的组数:\(x\ge0,z\ge1\)。\(......
  • Linux进程间通信 [补档-2023-07-27]
    Linux进程间通信10-1简介​在Linux下,进程之间相互独立,每个进程都有自己不同的用户地址空间。任何一个进程的全局变量在另一个进程中都看不到,所以进程和进程之间不能相互访问。如果非要交换数据则必须通过内核,在内核中开辟一块缓冲区。假设有两个进程AB,他们之间想......
  • Linux的信号管理 [补档-2023-07-30]
    信号11-1简介:​信号只是表示某个信号,不可以携带大量信息,信号需要满足特点的条件才会产生。是一种特别的通信手段。11-2信号机制:​假设有两个进程A,B,现在进程A给进程B发送信号,进程B在收到信号之前会执行自己的代码,当收到信号后,无论执行到了哪里,都要暂停执......
  • Linux文件IO之二 [补档-2023-07-21]
    8-5linux系统IO函数:open函数:​函数原型:intopen(constchar*pathname,intflags,mode_tmode);​功能:打开一个文件并返回文件描述符。与c库中的fopen差不多​参数:pathname:要打开的文件路径名。flags:打开文件的标志O_RDONLY(只读)O_WRONLY(只写)O_RD......
  • Linux的进程管理 [补档-2023-07-25]
    Linux进程管理9-1并发与并行:​并发:在同一个cpu上,并且在一个时间段时,同时运行多个程序。比如在1000毫秒内,我们有5个程序需要执行,所以我们可以将1000毫秒分为5个200毫秒,让每个程序都占用200毫秒的cpu使用权,这样在1000毫秒内就可以执行5个程序。​并行:大于等......
  • GDB调试程序 [补档-2023-07-19]
    gdb调试​它是gcc的调试工具,调试工具都能干什么就不多说了。7-1生成调试信息​在使用gcc编译c/c++的程序时,需要在编译命令中加入-g这一参数,它可以为你显示函数名,变量名等待。例如:gcc-gtest.c-otest​7-2启动gdb调试信息​启动指令:gdb可执行程序......
  • Linux文件IO之一 [补偿-2023-07-21]
    Linux文件IO8-1C标准库IO函数的工作流程​使用fopen函数打开一个文件,之后会返回一个FILE*fp指针,fp指针指向一个结构体,这个结构体是c标准io库中的一个结构体,这个结构体有三个重要的成员:文件描述符:描述符指向一个打开文件表,通过此表可以找到文件的inode表,通过对应的in......
  • MakeFile文件的使用 [补档-2023-07-13]
    makefile-gdb文件​可以在文件中指定那些文件可以先进行编译,那些文件可以后进行编译,那些文件可以重新编译。他可以自动化编译程序。。。。6-1makefile基本规则​如下:​目标:依赖​(tab)命令​规则三要素:目标:要生成的目标文件。依赖:目......
  • Linux系统的一些实用操作 [补档-2023-07-30]
    Linux的实用操作4-1.常用快捷键强制停止:当某些程序运行时,或者命令输入错误时,可以通过ctrl+c来强制结束当前的操作。退出或登出:当我们要退出某些用户时,或者要退出某些特殊的页面可以使用ctrl+d来进行。(vi/vim不可用)历史命令搜索:通过输入history命令来查看之前输入......