在网络编程中,UDP运用非常广泛。很多网络协议是基于UDP来实现的,如SNMP等。大家常常用到的局域网文件传输软件飞鸽传书也是基于UDP实现的。
本篇文章跟大家分享linux下UDP的使用和实现,主要介绍下sendto()和recvfrom()两个函数的使用,以及INADDR_ANY的说明,并在最后展示了一个经过自己测试可用的UDP Server和UDP Client的代码示例。
关于UDP数据报
UDP都是以数据报的形式进行发送和接收的,而TCP是以数据流的形式进行发送和接收的。数据报和数据流,这两者要区分开来。
头文件
函数原型
int recvfrom(int s, void *buf, int len, unsigned int flags, struct sockaddr *from, int *fromlen);
函数说明
sendto(),是把UDP数据报发给指定地址;recvfrom()是从指定地址接收UDP数据报。
参数说明
- \s: socket描述符。
- \buf: UDP数据报缓存地址。
- \len: UDP数据报长度。
- \flags: 该参数一般为0。
- \to: sendto()函数参数,struct sockaddr_in类型,指明UDP数据发往哪里报。
- \tolen: 对方地址长度,一般为:sizeof(struct sockaddr_in)。
- \from: recvfrom()函数参数,struct sockaddr 类型,指明UDP数据从哪里收。
- \fromlen:recvfrom()函数参数,struct sockaddr_in类型,指明从哪里接收UDP数据报。
函数返回值
对于sendto()函数,成功则返回实际传送出去的字符数,失败返回-1,错误原因存于errno 中。
对于recvfrom()函数,成功则返回接收到的字符数,失败则返回-1,错误原因存于errno中。
UDP Server和Client源码实例
UDP Server:
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#defineUDP_TEST_PORT50001
int main(int argC, char* arg[])
{
struct sockaddr_in addr;
int sockfd, len = 0;
int addr_len = sizeof(struct sockaddr_in);
char buffer[256];
/* 建立socket,注意必须是SOCK_DGRAM */
if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
perror ("socket");
exit(1);
}
/* 填写sockaddr_in 结构 */
bzero(&addr, sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(UDP_TEST_PORT);
addr.sin_addr.s_addr = htonl(INADDR_ANY) ;// 接收任意IP发来的数据
/* 绑定socket */
if (bind(sockfd, (struct sockaddr *)&addr, sizeof(addr))<0) {
perror("connect");
exit(1);
}
while(1) {
bzero(buffer, sizeof(buffer));
len = recvfrom(sockfd, buffer, sizeof(buffer), 0,
(struct sockaddr *)&addr ,&addr_len);
/* 显示client端的网络地址和收到的字符串消息 */
printf("Received a string from client %s, string is: %s\n",
inet_ntoa(addr.sin_addr), buffer);
/* 将收到的字符串消息返回给client端 */
sendto(sockfd,buffer, len, 0, (struct sockaddr *)&addr, addr_len);
}
return 0;
}
// ----------------------------------------------------------------------------
// End of udp_server.c
UDP Client:
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#defineUDP_TEST_PORT50001
#define UDP_SERVER_IP "127.0.0.1"
int main(int argC, char* arg[])
{
struct sockaddr_in addr;
int sockfd, len = 0;
int addr_len = sizeof(struct sockaddr_in);
char buffer[256];
/* 建立socket,注意必须是SOCK_DGRAM */
if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) {
perror("socket");
exit(1);
}
/* 填写sockaddr_in*/
bzero(&addr, sizeof(addr));
addr.sin_family = AF_INET;
addr.sin_port = htons(UDP_TEST_PORT);
addr.sin_addr.s_addr = inet_addr(UDP_SERVER_IP);
while(1) {
bzero(buffer, sizeof(buffer));
printf("Please enter a string to send to server: \n");
/* 从标准输入设备取得字符串*/
len = read(STDIN_FILENO, buffer, sizeof(buffer));
/* 将字符串传送给server端*/
sendto(sockfd, buffer, len, 0, (struct sockaddr *)&addr, addr_len);
/* 接收server端返回的字符串*/
len = recvfrom(sockfd, buffer, sizeof(buffer), 0,
(struct sockaddr *)&addr, &addr_len);
printf("Receive from server: %s\n", buffer);
}
return 0;
}
// ----------------------------------------------------------------------------
// End of udp_client.c
UDP Server:
01 | #include <sys/types.h> |
02 | #include <sys/socket.h> |
03 | #include <netinet/in.h> |
04 | #include <arpa/inet.h> |
05 | #include <unistd.h> |
06 | #include <stdlib.h> |
07 | #include <string.h> |
08 | #include <stdio.h> |
09 |
10 | #define UDP_TEST_PORT 50001 |
11 |
12 | int main( int argC, char * arg[]) |
13 | { |
14 | struct sockaddr_in addr; |
15 | int sockfd, len = 0; |
16 | int addr_len = sizeof ( struct sockaddr_in); |
17 | char buffer[256]; |
18 |
19 | /* 建立socket,注意必须是SOCK_DGRAM */ |
20 | if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { |
21 | perror ( "socket" ); |
22 | exit (1); |
23 | } |
24 |
25 | /* 填写sockaddr_in 结构 */ |
26 | bzero(&addr, sizeof (addr)); |
27 | addr.sin_family = AF_INET; |
28 | addr.sin_port = htons(UDP_TEST_PORT); |
29 | addr.sin_addr.s_addr = htonl(INADDR_ANY) ; // 接收任意IP发来的数据 |
30 |
31 | /* 绑定socket */ |
32 | if (bind(sockfd, ( struct sockaddr *)&addr, sizeof (addr))<0) { |
33 | perror ( "connect" ); |
34 | exit (1); |
35 | } |
36 |
37 | while (1) { |
38 | bzero(buffer, sizeof (buffer)); |
39 | len = recvfrom(sockfd, buffer, sizeof (buffer), 0, |
40 | ( struct sockaddr *)&addr ,&addr_len); |
41 | /* 显示client端的网络地址和收到的字符串消息 */ |
42 | printf ( "Received a string from client %s, string is: %s\n" , |
43 | inet_ntoa(addr.sin_addr), buffer); |
44 | /* 将收到的字符串消息返回给client端 */ |
45 | sendto(sockfd,buffer, len, 0, ( struct sockaddr *)&addr, addr_len); |
46 | } |
47 |
48 | return 0; |
49 | } |
50 |
51 | // ---------------------------------------------------------------------------- |
52 | // End of udp_server.c |
UDP Client:
01 | #include <sys/types.h> |
02 | #include <sys/socket.h> |
03 | #include <netinet/in.h> |
04 | #include <arpa/inet.h> |
05 | #include <unistd.h> |
06 | #include <stdlib.h> |
07 | #include <string.h> |
08 | #include <stdio.h> |
09 |
10 | #define UDP_TEST_PORT 50001 |
11 | #define UDP_SERVER_IP "127.0.0.1" |
12 |
13 | int main( int argC, char * arg[]) |
14 | { |
15 | struct sockaddr_in addr; |
16 | int sockfd, len = 0; |
17 | int addr_len = sizeof ( struct sockaddr_in); |
18 | char buffer[256]; |
19 |
20 | /* 建立socket,注意必须是SOCK_DGRAM */ |
21 | if ((sockfd = socket(AF_INET, SOCK_DGRAM, 0)) < 0) { |
22 | perror ( "socket" ); |
23 | exit (1); |
24 | } |
25 |
26 | /* 填写sockaddr_in*/ |
27 | bzero(&addr, sizeof (addr)); |
28 | addr.sin_family = AF_INET; |
29 | addr.sin_port = htons(UDP_TEST_PORT); |
30 | addr.sin_addr.s_addr = inet_addr(UDP_SERVER_IP); |
31 |
32 | while (1) { |
33 | bzero(buffer, sizeof (buffer)); |
34 |
35 | printf ( "Please enter a string to send to server: \n" ); |
36 |
37 | /* 从标准输入设备取得字符串*/ |
38 | len = read(STDIN_FILENO, buffer, sizeof (buffer)); |
39 |
40 | /* 将字符串传送给server端*/ |
41 | sendto(sockfd, buffer, len, 0, ( struct sockaddr *)&addr, addr_len); |
42 |
43 | /* 接收server端返回的字符串*/ |
44 | len = recvfrom(sockfd, buffer, sizeof (buffer), 0, |
45 | ( struct sockaddr *)&addr, &addr_len); |
46 | printf ( "Receive from server: %s\n" , buffer); |
47 | } |
48 |
49 | return 0; |
50 | } |
51 |
52 | // ---------------------------------------------------------------------------- |
53 | // End of udp_client.c |
上述代码是经过验证可用的。