4.UDP
UDP沒有傳遞保證機制,如果傳輸中數據丟失,協議不會有任何的檢測或提示。
這樣的好處是傳輸的數據是持續的,此外它是無連接的傳輸,比如實時視頻時,如果采用TCP,中途有一點點數據出錯都會卡住,進行等待,產生延時。加入使用UDP,盡管有少量的丟幀,但數據是實時的。
4.1 UDP流程圖
4.2 UDP步驟分析
從流程圖可以看出,UDP比TCP的步驟少多了。
- 服務器端:
a. 創建socket
1 sock_fd = socket(AF_INET, SOCK_DGRAM, 0);//AF_INET:IPV4;SOCK_DGRAM:UDP
2 if (-1 == sock_fd)
3 {
4 fprintf(stderr,"socket error:%s\\n\\a", strerror(errno));
5 exit(1);
6 }
協議族改成SOCK_DGRAM。
b. 設置socket
1 memset(&server_addr, 0, sizeof(struct sockaddr_in));//clear
2 server_addr.sin_family = AF_INET;
3 server_addr.sin_addr.s_addr = htonl(INADDR_ANY);//INADDR_ANY:This machine all IP
4 server_addr.sin_port = htons(PORT_NUMBER);
和前面的TCP設置還是一樣的。
c. 綁定socket
1 ret = bind(sock_fd, (struct sockaddr *)(&server_addr), sizeof(struct sockaddr));
2 if(-1 == ret)
3 {
4 fprintf(stderr,"bind error:%s\\n\\a", strerror(errno));
5 close(sock_fd);
6 exit(1);
7 }
綁定的操作也沒有變。
d. 接收數據
1 addr_len = sizeof(struct sockaddr);
2 recv_len = recvfrom(sock_fd, recv_buf, 999, 0, (struct sockaddr *)&server_addr, &addr_len);
3 if (recv_len <= 0)
4 {
5 fprintf(stderr, "recvfrom error:%s\\n\\a", strerror(errno));
6 close(sock_fd);
7 exit(1);
8 }
9 else
10 {
11 recv_buf[recv_len] = '\\0';
12 printf("Get msg from client%d: %s\\n", client_num, recv_buf);
13 }
所需要頭文件 :
#include
#include
函數格式 :
int recvfrom(int sockfd, char FAR *buf, int len, int flags, struct sockaddr FAR *from, int FAR *fromlen);
函數功能 :
從套接字上接收一個數據報并保存源地址;
sockfd:標識一個已連接套接字的描述符
buf:接收數據緩沖區
len:接收數據緩沖區長度
flags:調用操作方式,由以下零個或多個組成
flags 說明 recv send MSG_DONTROUTE 繞過路由表查找 ? MSG_DONTWAIT 僅本操作非阻塞 ? ? MSG_OOB 發送或接收帶外數據 ? ? MSG_PEEK 窺看外來消息 ? MSG_WAITALL 等待所有數據 ? from:(可選)指針,指向裝有源地址的緩沖區
fromlen:(可選)指針,指向from緩沖區長度值
返回值 :
若成功,返回讀入的字節數,否則返回0;
e. 關閉
1 close(sock_fd);
2 exit(0);
- 客戶機端:
a. 創建socket
1 sock_fd = socket(AF_INET, SOCK_DGRAM, 0);//AF_INET:IPV4;SOCK_DGRAM:UDP
2 if (-1 == sock_fd)
3 {
4 fprintf(stderr,"socket error:%s\\n\\a", strerror(errno));
5 exit(1);
6 }
協議族改成SOCK_DGRAM。
b. 設置socket
1 memset(&server_addr, 0, sizeof(struct sockaddr_in));//clear
2 server_addr.sin_family = AF_INET;
3 server_addr.sin_port = htons(PORT_NUMBER);
4 ret = inet_aton(argv[1], &server_addr.sin_addr);
5 if(0 == ret)
6 {
7 fprintf(stderr,"server_ip error.\\n");
8 close(sock_fd);
9 exit(1);
10 }
c. 發送數據
1 addr_len = sizeof(struct sockaddr);
2 send_len = sendto(sock_fd, send_buf, strlen(send_buf), 0, (const struct sockaddr *)&server_addr, addr_len);
3 if (send_len <= 0)
4 {
5 fprintf(stderr,"send error:%s\\n\\a", strerror(errno));
6 close(sock_fd);
7 exit(1);
8 }
所需要頭文件 :
#include
#include
函數格式 :
int sendto(int sockfd, char FAR *buf, int len, int flags, struct sockaddr FAR *to, int FAR *tolen);
函數功能 :
向一指定目的地發送數據;
sockfd:一個標識套接字的描述字
buf:發送數據緩沖區
len:發送數據緩沖區長度
flags:調用方式標志位
to:(可選)指針,指向目的的套接字的地址
tolen:目的套接字地址的長度
返回值 :
若成功,返回發送的字節數,如果連接已中止,返回0,如果發生錯誤,返回-1;
d. 關閉
1 close(sock_fd);
2 exit(0);
UDP傳輸的客戶端少了connect(),原本該在connect()函數里傳入服務器地址相關信息,現在變成了在sendto()里傳入。
4.3 UDP完整代碼
1/*
2* udp_server.c
3# Copyright (C) 2017 hceng,
9#include
10#include
11#include
12#include
13#include
14#include
15#include
16#include
17#include
18#include
19#include
20
21#define PORT_NUMBER 8888
22
23/* socket->bind->recvfrom/sendto->close */
24
25int main(int argc, char **argv)
26{
27 int sock_fd;
28 struct sockaddr_in server_addr;
29 struct sockaddr_in client_addr;
30 int ret;
31 int addr_len;
32 int recv_len;
33 unsigned char recv_buf[1000];
34
35 /* socket */
36 sock_fd = socket(AF_INET, SOCK_DGRAM, 0);//AF_INET:IPV4;SOCK_DGRAM:UDP
37 if (-1 == sock_fd)
38 {
39 fprintf(stderr,"socket error:%s\\n\\a", strerror(errno));
40 exit(1);
41 }
42
43 /* set sockaddr_in parameter*/
44 memset(&server_addr, 0, sizeof(struct sockaddr_in));//clear
45 server_addr.sin_family = AF_INET;
46 server_addr.sin_addr.s_addr = htonl(INADDR_ANY);//INADDR_ANY:This machine all IP
47 server_addr.sin_port = htons(PORT_NUMBER);
48
49 /* bind */
50 ret = bind(sock_fd, (struct sockaddr *)(&server_addr), sizeof(struct sockaddr));
51 if(-1 == ret)
52 {
53 fprintf(stderr,"bind error:%s\\n\\a", strerror(errno));
54 close(sock_fd);
55 exit(1);
56 }
57
58 while (1)
59 {
60 /* recvfrom */
61 addr_len = sizeof(struct sockaddr);
62 recv_len = recvfrom(sock_fd, recv_buf, 999, 0, (struct sockaddr *)&client_addr, &addr_len);
63 if (recv_len <= 0)
64 {
65 fprintf(stderr, "recvfrom error:%s\\n\\a", strerror(errno));
66 close(sock_fd);
67 exit(1);
68 }
69 else
70 {
71 recv_buf[recv_len] = '\\0';
72 printf("Get msg from client:%s: %s\\n", inet_ntoa(client_addr.sin_addr), recv_buf);
73 }
74 }
75
76 /* close */
77 close(sock_fd);
78 exit(0);
79}
1/*
2* udp_client.c
3# Copyright (C) 2017 hceng,
9#include
10#include
11#include
12#include
13#include
14#include
15#include
16#include
17#include
18#include
19
20#define PORT_NUMBER 8888
21
22/* socket->bind->recvfrom/sendto->close */
23int main(int argc, char *argv[])
24{
25 int sock_fd;
26 struct sockaddr_in server_addr;
27 int ret;
28 unsigned char send_buf[1000];
29 int send_len;
30 int addr_len;
31
32 if(argc != 2)
33 {
34 fprintf(stderr, "Usage:%s hostname\\n\\a", argv[0]);
35 exit(1);
36 }
37
38 /* socket */
39 sock_fd = socket(AF_INET, SOCK_DGRAM, 0);//AF_INET:IPV4;SOCK_DGRAM:UDP
40 if (-1 == sock_fd)
41 {
42 fprintf(stderr,"socket error:%s\\n\\a", strerror(errno));
43 exit(1);
44 }
45
46 /* set sockaddr_in parameter*/
47 memset(&server_addr, 0, sizeof(struct sockaddr_in));//clear
48 server_addr.sin_family = AF_INET;
49 server_addr.sin_port = htons(PORT_NUMBER);
50 ret = inet_aton(argv[1], &server_addr.sin_addr);
51 if(0 == ret)
52 {
53 fprintf(stderr,"server_ip error.\\n");
54 close(sock_fd);
55 exit(1);
56 }
57
58 while (1)
59 {
60 if (fgets(send_buf, 999, stdin))
61 {
62 /* sendto */
63 addr_len = sizeof(struct sockaddr);
64 send_len = sendto(sock_fd, send_buf, strlen(send_buf), 0, \\
65(const struct sockaddr *)&server_addr, addr_len);
66 if (send_len <= 0)
67 {
68 fprintf(stderr,"send error:%s\\n\\a", strerror(errno));
69 close(sock_fd);
70 exit(1);
71 }
72 }
73 }
74
75 /* close */
76 close(sock_fd);
77 exit(0);
78}
4.4 測試結果
和前面TCP測試方式一樣,先在Ubuntu主機上交叉編譯服務器端代碼,再在Ubuntu主機上編譯客戶端代碼。
在開發板上運行服務器端代碼,在Ubuntu主機先啟動tmux分屏,再分別運行客戶端代碼。
-
服務器端
-
客戶機端
-
TCP
+關注
關注
8文章
1372瀏覽量
79143 -
UDP
+關注
關注
0文章
326瀏覽量
33993 -
網絡通信
+關注
關注
4文章
808瀏覽量
29848 -
網絡編程
+關注
關注
0文章
72瀏覽量
10085
發布評論請先 登錄
相關推薦
評論