在網(wǎng)絡(luò)傳輸協(xié)議中,TCP協(xié)議提供的是一種可靠的,復(fù)雜的,面向連接的數(shù)據(jù)流(SOCK_STREAM)傳輸服務(wù),它通過三段式握手過程建立連接。TCP有一種“重傳確認(rèn)”機(jī)制,即接收端收到數(shù)據(jù)后要發(fā)出一個(gè)肯定確認(rèn)的信號(hào),發(fā)送端如果收到接收端肯定確認(rèn)的信號(hào),就會(huì)繼續(xù)發(fā)送其他的數(shù)據(jù),如果沒有,它就會(huì)重新發(fā)送。
相對(duì)而言,UDP協(xié)議則是一種無連接的,不可靠的數(shù)據(jù)報(bào)(SOCK_DGRAM)傳輸服務(wù)。使用UDP套接口不用建立連接,服務(wù)端在調(diào)用socket()生成一個(gè)套接字并調(diào)用bind()綁定端口后就可以進(jìn)行通信(recvfrom函數(shù)和sendto函數(shù))了;客戶端在用socket()生成一個(gè)套接字后就可以向服務(wù)端地址發(fā)送和接收數(shù)據(jù)了。
此處需要特別注意:TCP使用的是流套接字(SOCK_STREAM),UDP使用的是數(shù)據(jù)報(bào)套接字(SOCK_DGRAM)
UDP套接字編程范例:
server端代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
|
/************************************************************************* > File Name: server.c > Author: SongLee ************************************************************************/ #include<sys/types.h> #include<sys/socket.h> #include<unistd.h> #include<netinet/in.h> #include<arpa/inet.h> #include<stdio.h> #include<stdlib.h> #include<errno.h> #include<netdb.h> #include<stdarg.h> #include<string.h> #define SERVER_PORT 8000 #define BUFFER_SIZE 1024 #define FILE_NAME_MAX_SIZE 512 int main() { /* 創(chuàng)建UDP套接口 */ struct sockaddr_in server_addr; bzero(&server_addr, sizeof (server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = htonl(INADDR_ANY); server_addr.sin_port = htons(SERVER_PORT); /* 創(chuàng)建socket */ int server_socket_fd = socket(AF_INET, SOCK_DGRAM, 0); if (server_socket_fd == -1) { exit (1); } /* 綁定套接口 */ if (-1 == (bind(server_socket_fd,( struct sockaddr*)&server_addr, sizeof (server_addr)))) { perror ( "Server Bind Failed:" ); exit (1); } /* 數(shù)據(jù)傳輸 */ while (1) { /* 定義一個(gè)地址,用于捕獲客戶端地址 */ struct sockaddr_in client_addr; socklen_t client_addr_length = sizeof (client_addr); /* 接收數(shù)據(jù) */ char buffer[BUFFER_SIZE]; bzero(buffer, BUFFER_SIZE); if (recvfrom(server_socket_fd, buffer, BUFFER_SIZE,0,( struct sockaddr*)&client_addr, &client_addr_length) == -1) { perror ( "Receive Data Failed:" ); exit (1); } /* 從buffer中拷貝出file_name */ char file_name[FILE_NAME_MAX_SIZE+1]; bzero(file_name,FILE_NAME_MAX_SIZE+1); strncpy (file_name, buffer, strlen (buffer)>FILE_NAME_MAX_SIZE?FILE_NAME_MAX_SIZE: strlen (buffer)); printf ( "%s\n" , file_name); } close(server_socket_fd); return 0; } |
client端代碼如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
|
/************************************************************************* > File Name: client.c > Author: SongLee ************************************************************************/ #include<sys/types.h> #include<sys/socket.h> #include<unistd.h> #include<netinet/in.h> #include<arpa/inet.h> #include<stdio.h> #include<stdlib.h> #include<errno.h> #include<netdb.h> #include<stdarg.h> #include<string.h> #define SERVER_PORT 8000 #define BUFFER_SIZE 1024 #define FILE_NAME_MAX_SIZE 512 int main() { /* 服務(wù)端地址 */ struct sockaddr_in server_addr; bzero(&server_addr, sizeof (server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_addr.s_addr = inet_addr( "127.0.0.1" ); server_addr.sin_port = htons(SERVER_PORT); /* 創(chuàng)建socket */ int client_socket_fd = socket(AF_INET, SOCK_DGRAM, 0); if (client_socket_fd < 0) { perror ( "Create Socket Failed:" ); exit (1); } /* 輸入文件名到緩沖區(qū) */ char file_name[FILE_NAME_MAX_SIZE+1]; bzero(file_name, FILE_NAME_MAX_SIZE+1); printf ( "Please Input File Name On Server:\t" ); scanf ( "%s" , file_name); char buffer[BUFFER_SIZE]; bzero(buffer, BUFFER_SIZE); strncpy (buffer, file_name, strlen (file_name)>BUFFER_SIZE?BUFFER_SIZE: strlen (file_name)); /* 發(fā)送文件名 */ if (sendto(client_socket_fd, buffer, BUFFER_SIZE,0,( struct sockaddr*)&server_addr, sizeof (server_addr)) < 0) { perror ( "Send File Name Failed:" ); exit (1); } close(client_socket_fd); return 0; } |
讀者可以參考對(duì)比前一篇:Linux網(wǎng)絡(luò)編程之socket文件傳輸示例,注意UDP和TCP工作流程的對(duì)比。以加深對(duì)該程序原理的理解。