UDP协议 sendto 和 recvfrom 浅析与示例

UDP协议 sendto 和 recvfrom 浅析与示例UDP(userdatagramprotocol)用户数据报协议,属于传输层。UDP是面向非连接的协议,它不与对方建立连接,而是直接把数据报发给对方。UDP无需建立类如三次握手的连接,使得通信效

大家好,又见面了,我是你们的朋友全栈君。

  UDP(user datagram protocol)用户数据报协议,属于传输层。

  UDP是面向非连接的协议,它不与对方建立连接,而是直接把数据报发给对方。UDP无需建立类如三次握手的连接,使得通信效率很高。因此UDP适用于一次传输数据量很少、对可靠性要求不高的或对实时性要求高的应用场景。

UDP示意图

  UDP通信的过程如图所示:

    服务端:

      (1)使用函数socket(),生成套接字文件描述符;

      (2)通过struct sockaddr_in 结构设置服务器地址和监听端口;

      (3)使用bind() 函数绑定监听端口,将套接字文件描述符和地址类型变量(struct sockaddr_in )进行绑定;

      (4)接收客户端的数据,使用recvfrom() 函数接收客户端的网络数据;

      (5)向客户端发送数据,使用sendto() 函数向服务器主机发送数据;

      (6)关闭套接字,使用close() 函数释放资源;

    客户端:

      (1)使用socket(),生成套接字文件描述符;

      (2)通过struct sockaddr_in 结构设置服务器地址和监听端口;

      (3)向服务器发送数据,sendto() ;

      (4)接收服务器的数据,recvfrom() ;

      (5)关闭套接字,close() ;

  (关于 sockaddr sockaddr_in 的区别,可参考:https://blog.csdn.net/qingzhuyuxian/article/details/79736821

 

sendto()

1 1 int sendto(int s, const void *buf, int len, unsigned int flags, 2         const struct sockaddr *to, int tolen);

  返回值说明:

    成功则返回实际传送出去的字符数,失败返回-1,错误原因会存于errno 中。

  参数说明:

    s:      socket描述符;
    buf:  UDP数据报缓存区(包含待发送数据);
    len:   UDP数据报的长度;
    flags:调用方式标志位(一般设置为0);
    to:  指向接收数据的主机地址信息的结构体(sockaddr_in需类型转换);
    tolen:to所指结构体的长度;

recvfrom()

1 int recvfrom(int s, void *buf, int len, unsigned int flags, 2           struct sockaddr *from, int *fromlen); 

  返回值说明:

    成功则返回实际接收到的字符数,失败返回-1,错误原因会存于errno 中。

  参数说明:

    s:          socket描述符;
    buf:       UDP数据报缓存区(包含所接收的数据); 
    len:       缓冲区长度。 
    flags:    调用操作方式(一般设置为0)。 
    from:     指向发送数据的客户端地址信息的结构体(sockaddr_in需类型转换);
    fromlen:指针,指向from结构体长度值。

 

示例代码

服务端

<span role="heading" aria-level="2">UDP协议 sendto 和 recvfrom 浅析与示例
<span role="heading" aria-level="2">UDP协议 sendto 和 recvfrom 浅析与示例

 1 #include <unistd.h>
 2 #include <string.h>
 3 #include <stdio.h>
 4 #include <iostream>
 5 #include <netinet/in.h>
 6 #include <sys/socket.h>
 7 #include <sys/types.h>
 8 #include <arpa/inet.h>
 9 
10 #define MAXLINE 4096
11 #define UDPPORT 8001
12 #define SERVERIP "192.168.255.129"
13 
14 using namespace std;
15 
16 int main(){
17     int serverfd;
18     unsigned int server_addr_length, client_addr_length;
19     char recvline[MAXLINE];
20     char sendline[MAXLINE];
21     struct sockaddr_in serveraddr , clientaddr;
22 
23     // 使用函数socket(),生成套接字文件描述符;
24     if( (serverfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0 ){
25         perror("socket() error");
26         exit(1);
27     }
28 
29     // 通过struct sockaddr_in 结构设置服务器地址和监听端口;
30     bzero(&serveraddr,sizeof(serveraddr));
31     serveraddr.sin_family = AF_INET;
32     serveraddr.sin_addr.s_addr = htonl(INADDR_ANY);
33     serveraddr.sin_port = htons(UDPPORT);
34     server_addr_length = sizeof(serveraddr);
35 
36     // 使用bind() 函数绑定监听端口,将套接字文件描述符和地址类型变量(struct sockaddr_in )进行绑定;
37     if( bind(serverfd, (struct sockaddr *) &serveraddr, server_addr_length) < 0){
38         perror("bind() error");
39         exit(1); 
40     }
41     
42     // 接收客户端的数据,使用recvfrom() 函数接收客户端的网络数据;
43     client_addr_length = sizeof(sockaddr_in);
44     int recv_length = 0;
45     recv_length = recvfrom(serverfd, recvline, sizeof(recvline), 0, (struct sockaddr *) &clientaddr, &client_addr_length);
46     cout << "recv_length = "<< recv_length <<endl;
47     cout << recvline << endl;
48     
49     // 向客户端发送数据,使用sendto() 函数向服务器主机发送数据;
50     int send_length = 0;
51     sprintf(sendline, "hello client !");
52     send_length = sendto(serverfd, sendline, sizeof(sendline), 0, (struct sockaddr *) &clientaddr, client_addr_length);
53     if( send_length < 0){
54         perror("sendto() error");
55         exit(1);
56     }
57     cout << "send_length = "<< send_length <<endl;
58     
59     //关闭套接字,使用close() 函数释放资源;
60     close(serverfd);
61 
62     return 0;
63 }

View Code

 

客户端

<span role="heading" aria-level="2">UDP协议 sendto 和 recvfrom 浅析与示例
<span role="heading" aria-level="2">UDP协议 sendto 和 recvfrom 浅析与示例

 1 #include <unistd.h>
 2 #include <string.h>
 3 #include <stdio.h>
 4 #include <iostream>
 5 #include <netinet/in.h>
 6 #include <sys/socket.h>
 7 #include <sys/types.h>
 8 #include <arpa/inet.h>
 9 
10 #define MAXLINE 4096
11 #define UDPPORT 8001
12 #define SERVERIP "192.168.255.129"
13 
14 using namespace std;
15 
16 int main(){
17     int confd;
18     unsigned int addr_length;
19     char recvline[MAXLINE];
20     char sendline[MAXLINE];
21     struct sockaddr_in serveraddr;
22 
23     // 使用socket(),生成套接字文件描述符;
24     if( (confd = socket(AF_INET, SOCK_DGRAM, 0)) < 0 ){
25         perror("socket() error");
26         exit(1);
27     }
28 
29     //通过struct sockaddr_in 结构设置服务器地址和监听端口;
30     bzero(&serveraddr, sizeof(serveraddr));
31     serveraddr.sin_family = AF_INET;
32     serveraddr.sin_addr.s_addr = inet_addr(SERVERIP);
33     serveraddr.sin_port = htons(UDPPORT);
34     addr_length = sizeof(serveraddr);
35 
36     // 向服务器发送数据,sendto() ;
37     int send_length = 0;
38     sprintf(sendline,"hello server!");
39     send_length = sendto(confd, sendline, sizeof(sendline), 0, (struct sockaddr *) &serveraddr, addr_length);
40     if(send_length < 0 ){
41         perror("sendto() error");
42         exit(1);
43     }
44     cout << "send_length = " << send_length << endl;
45 
46     // 接收服务器的数据,recvfrom() ;
47     int recv_length = 0;
48     recv_length = recvfrom(confd, recvline, sizeof(recvline), 0, (struct sockaddr *) &serveraddr, &addr_length);
49     cout << "recv_length = " << recv_length <<endl;
50     cout << recvline << endl; 
51 
52     // 关闭套接字,close() ;
53     close(confd);
54 
55     return 0;
56 }

View Code

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/155014.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • python爬虫scrapy框架_nodejs爬虫框架

    python爬虫scrapy框架_nodejs爬虫框架叮铃铃!叮铃铃!老师:“小明你的梦想是什么?”,沉思了一下小明:“额额 我想有车有房,自己当老板,媳妇貌美如花,还有一个当官的兄弟”老师:“北宋有一个人和你一样···”;哈喽!大家好!请叫我布莱恩·奥复托·杰森张;爬虫部分!一提到爬虫,好多人先想到python没错就是那个py交易的那个,这货所为是什么都能干上九天揽月下五洋捉鳖无处不出现它的身影鄙人对它也是不得不折

    2026年1月15日
    4
  • 网管必备工具_ps功能介绍与工具使用视频

    网管必备工具_ps功能介绍与工具使用视频http://book.51cto.com/art/200903/116214.htm 转载于:https://blog.51cto.com/netsecing/163178

    2022年10月5日
    3
  • scratch编程一款节奏小游戏「建议收藏」

    scratch编程一款节奏小游戏「建议收藏」今天小恐龙来教大家做一款好玩的小游戏:这个游戏本恐龙没有加音乐,因为箭头落下的速度会根据我们的命中率来改变,按中的越多速度就越快,漏的越多速度就越慢,有上下限!首先画出轨道,粗细随意:轨道直接设置成移到(0,0)就可以了,没有别的程序然后是箭头:变量分值和速度是隐藏起来的,我们在玩的时候是看不到的,速度下限是5,上限是10,不然太快了反应不过来!箭头克隆体程序:最后是判定线的程序:如果是觉得没有音乐太单调的话,可以通过创建一个链表来储存音符。再新建一个变量,每按中一个箭头就将

    2022年6月16日
    55
  • webstorm2021永久激活【2021.10最新】

    (webstorm2021永久激活)这是一篇idea技术相关文章,由全栈君为大家提供,主要知识点是关于2021JetBrains全家桶永久激活码的内容IntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,下面是详细链接哦~https://javaforall.net/100143.html1435QFILVV-eyJsaWN…

    2022年3月30日
    777
  • python为什么叫爬虫_检测安全

    python为什么叫爬虫_检测安全前言周一一早网管收到来自阿里云的一堆警告,发现我们维护的一个网站下有数十个被挂马的文件。网管直接关了vsftpd,然后把警告导出邮件给我们。取出部分大致如下:服务器IP/名称木马文件路径更新时间木马类型状态(全部)*.*.*.*/path/*144.gif2017/8/75:53Webshell待处理*.*.*.*/path/*…

    2022年9月29日
    5
  • Redis之压缩列表ziplist

    Redis是基于内存的nosql,有些场景下为了节省内存redis会用“时间”换“空间”。ziplist就是很典型的例子。ziplist是list键、hash键以及zset键的底层实现之一(3.0之后list键已经不直接用ziplist和linkedlist作为底层实现了,取而代之的是quicklist)这些键的常规底层实现如下:list键:双向链表 hash键:字典di…

    2022年4月9日
    81

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号