基于UDP的服务器端和客户端

原创 2016年06月02日 08:07:58
前面的文章中我们给出了几个TCP的例子,对于UDP而言,只要能理解前面的内容,实现并非难事。

UDP中的服务器端和客户端没有连接

UDP不像TCP,无需在连接状态下交换数据,因此基于UDP的服务器端和客户端也无需经过连接过程。也就是说,不必调用 listen() 和 accept() 函数。UDP中只有创建套接字的过程和数据交换的过程。

UDP服务器端和客户端均只需1个套接字

TCP中,套接字是一对一的关系。如要向10个客户端提供服务,那么除了负责监听的套接字外,还需要创建10套接字。但在UDP中,不管是服务器端还是客户端都只需要1个套接字。之前解释UDP原理的时候举了邮寄包裹的例子,负责邮寄包裹的快递公司可以比喻为UDP套接字,只要有1个快递公司,就可以通过它向任意地址邮寄包裹。同样,只需1个UDP套接字就可以向任意主机传送数据。

基于UDP的接收和发送函数

创建好TCP套接字后,传输数据时无需再添加地址信息,因为TCP套接字将保持与对方套接字的连接。换言之,TCP套接字知道目标地址信息。但UDP套接字不会保持连接状态,每次传输数据都要添加目标地址信息,这相当于在邮寄包裹前填写收件人地址。

发送数据使用 sendto() 函数:
  1. ssize_t sendto(int sock, void *buf, size_t nbytes, int flags, struct sockaddr *to, socklen_t addrlen); //Linux
  2. int sendto(SOCKET sock, const char *buf, int nbytes, int flags, const struct sockadr *to, int addrlen); //Windows
Linux和Windows下的 sendto() 函数类似,下面是详细参数说明:
  • sock:用于传输UDP数据的套接字;
  • buf:保存待传输数据的缓冲区地址;
  • nbytes:带传输数据的长度(以字节计);
  • flags:可选项参数,若没有可传递0;
  • to:存有目标地址信息的 sockaddr 结构体变量的地址;
  • addrlen:传递给参数 to 的地址值结构体变量的长度。

UDP 发送函数 sendto() 与TCP发送函数 write()/send() 的最大区别在于,sendto() 函数需要向他传递目标地址信息。

接收数据使用 recvfrom() 函数:
  1. ssize_t recvfrom(int sock, void *buf, size_t nbytes, int flags, struct sockadr *from, socklen_t *addrlen); //Linux
  2. int recvfrom(SOCKET sock, char *buf, int nbytes, int flags, const struct sockaddr *from, int *addrlen); //Windows
由于UDP数据的发送端不不定,所以 recvfrom() 函数定义为可接收发送端信息的形式,具体参数如下:
  • sock:用于接收UDP数据的套接字;
  • buf:保存接收数据的缓冲区地址;
  • nbytes:可接收的最大字节数(不能超过buf缓冲区的大小);
  • flags:可选项参数,若没有可传递0;
  • from:存有发送端地址信息的sockaddr结构体变量的地址;
  • addrlen:保存参数 from 的结构体变量长度的变量地址值。

基于UDP的回声服务器端/客户端

下面结合之前的内容实现回声客户端。需要注意的是,UDP不同于TCP,不存在请求连接和受理过程,因此在某种意义上无法明确区分服务器端和客户端,只是因为其提供服务而称为服务器端,希望各位读者不要误解。

下面给出Windows下的代码,Linux与此类似,不再赘述。

服务器端 server.cpp:
  1. #include <stdio.h>
  2. #include <winsock2.h>
  3. #pragma comment (lib, "ws2_32.lib") //加载 ws2_32.dll
  4. #define BUF_SIZE 100
  5. int main(){
  6. WSADATA wsaData;
  7. WSAStartup( MAKEWORD(2, 2), &wsaData);
  8. //创建套接字
  9. SOCKET sock = socket(AF_INET, SOCK_DGRAM, 0);
  10. //绑定套接字
  11. sockaddr_in servAddr;
  12. memset(&servAddr, 0, sizeof(servAddr)); //每个字节都用0填充
  13. servAddr.sin_family = PF_INET; //使用IPv4地址
  14. servAddr.sin_addr.s_addr = htonl(INADDR_ANY); //自动获取IP地址
  15. servAddr.sin_port = htons(1234); //端口
  16. bind(sock, (SOCKADDR*)&servAddr, sizeof(SOCKADDR));
  17. //接收客户端请求
  18. SOCKADDR clntAddr; //客户端地址信息
  19. int nSize = sizeof(SOCKADDR);
  20. char buffer[BUF_SIZE]; //缓冲区
  21. while(1){
  22. int strLen = recvfrom(sock, buffer, BUF_SIZE, 0, &clntAddr, &nSize);
  23. sendto(sock, buffer, strLen, 0, &clntAddr, nSize);
  24. }
  25. closesocket(sock);
  26. WSACleanup();
  27. return 0;
  28. }
代码说明:
1) 第12行代码在创建套接字时,向 socket() 第二个参数传递 SOCK_DGRAM,以指明使用UDP协议。

2) 第18行代码中使用htonl(INADDR_ANY)来自动获取IP地址。

利用常数 INADDR_ANY 自动获取IP地址有一个明显的好处,就是当软件安装到其他服务器或者服务器IP地址改变时,不用再更改源码重新编译,也不用在启动软件时手动输入。而且,如果一台计算机中已分配多个IP地址(例如路由器),那么只要端口号一致,就可以从不同的IP地址接收数据。所以,服务器中优先考虑使用INADDR_ANY;而客户端中除非带有一部分服务器功能,否则不会采用。

客户端 client.cpp:
  1. #include <stdio.h>
  2. #include <WinSock2.h>
  3. #pragma comment(lib, "ws2_32.lib") //加载 ws2_32.dll
  4. #define BUF_SIZE 100
  5. int main(){
  6. //初始化DLL
  7. WSADATA wsaData;
  8. WSAStartup(MAKEWORD(2, 2), &wsaData);
  9. //创建套接字
  10. SOCKET sock = socket(PF_INET, SOCK_DGRAM, 0);
  11. //服务器地址信息
  12. sockaddr_in servAddr;
  13. memset(&servAddr, 0, sizeof(servAddr)); //每个字节都用0填充
  14. servAddr.sin_family = PF_INET;
  15. servAddr.sin_addr.s_addr = inet_addr("127.0.0.1");
  16. servAddr.sin_port = htons(1234);
  17. //不断获取用户输入并发送给服务器,然后接受服务器数据
  18. sockaddr fromAddr;
  19. int addrLen = sizeof(fromAddr);
  20. while(1){
  21. char buffer[BUF_SIZE] = {0};
  22. printf("Input a string: ");
  23. gets(buffer);
  24. sendto(sock, buffer, strlen(buffer), 0, (struct sockaddr*)&servAddr, sizeof(servAddr));
  25. int strLen = recvfrom(sock, buffer, BUF_SIZE, 0, &fromAddr, &addrLen);
  26. buffer[strLen] = 0;
  27. printf("Message form server: %s\n", buffer);
  28. }
  29. closesocket(sock);
  30. WSACleanup();
  31. return 0;
  32. }
先运行 server,再运行 client,client 输出结果为:

Input a string: C语言中文网
Message form server: C语言中文网
Input a string: c.biancheng.net Founded in 2012
Message form server: c.biancheng.net Founded in 2012
Input a string:


从代码中可以看出,server.cpp 中没有使用 listen() 函数,client.cpp 中也没有使用 connect() 函数,因为 UDP 不需要连接。
版权声明:本文为芝麻软件工作室原创文章,未经芝麻软件工作室允许不得转载。

相关文章推荐

UDP服务器端和客户端程序设计

实验三 UDP服务器端程序设计 一、实验目的 学习和掌握Linux下的UDP服务器基本原理和基本编程方法,体会与TCP的区别,TCP编程:http://blog.csdn.net/yueguang...

Android网络编程TCP、UDP(一)

一、TCP与UDP简介TCP和UDP都属于TCP/IP参考模型中传输层的协议,且都是基于网际互联层IP协议。一位大神作了一个很形象的比喻:TCP和UDP使用IP协议从一个网络传送数据包到另一个网络。把...
  • a10615
  • a10615
  • 2016-09-01 02:21
  • 3046

Android网络编程TCP、UDP(二)

先对上一遍的工具类,补充两点: 1、Client关闭异常 如果没有连接host就调用close()的话,会导致NullPointException,因为mInputStream为null。虽然so...
  • a10615
  • a10615
  • 2016-09-04 00:23
  • 2736

【windows socket+UDP服务器客户端】

Windows Socket+TCP服务器客户端      Winsock是 Windows下套接字标准。          1.UDP socket编程:         ...

AI大行其道,你准备好了吗?—谨送给徘徊于转行AI的程序员

前言  近年来,随着 Google 的 AlphaGo 打败韩国围棋棋手李世乭之后,机器学习尤其是深度学习的热潮席卷了整个IT界。所有的互联网公司,尤其是 Google 微软,百度,腾讯等巨头,无不在...

用UDP协议实现最简单的“聊天室”功能

小弟新人一枚,刚刚学习了网络编程的基础,就试着用UDP协议实现一个简单的“聊天室”,在写代码过程中也是各种出错,还好最终是实现了各种最基本的功能,在此记录一下当时写代码时的各种问题,希望跟我一样的新人...

UDP服务器和客户端交互

实验三 UDP服务器端程序设计 一、实验目的 学习和掌握Linux下的UDP服务器基本原理和基本编程方法,体会与TCP的区别,TCP编程:http://blog.csdn.net/yueg...

unix文件系统模拟-操作系统课程设计

本周进行操作系统课程设计,在很多的题目中选了个unix文件系统模拟,主要就是操作结构与文件。 为了方便,文件系统结构如下: Super block   --  Block bitmap  -- I...

最简单服务器程序UDP(Linux)

最简单服务器程序UDP(Linux)

UDP并发服务器模型 一

摘要: 本文将讨论UDP的并发实现机制。给出了两种实现方法。第一种是最为常见的,TFTP传输的方式。 第二种是笔者无聊时自己编写,功能难免会比较简单、也会有许多不足...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)