UDP協議是User Datagram Protocol的簡稱,是一種無連接的協議,每個數據報都是一個獨立的信息,包括完整的源地址或目的地址,它在網絡上以任何可能的路徑傳往目的地,因此能否到達目的地,到達目的地的時間以及內容的正確性都是不能被保證的。
既然有了保證可靠傳輸的TCP協議,為什么還要非可靠傳輸的UDP協議呢?主要的原因有兩個。一是可靠的傳輸是要付出代價的,對數據內容正確性的檢驗必然占用計算機的處理時間和網絡的帶寬,因此TCP傳輸的效率不如UDP高。二是在許多應用中并不需要保證嚴格的傳輸可靠性,比如視頻會議系統,并不要求音頻視頻數據絕對的正確,只要保證連貫性就可以了,這種情況下顯然使用UDP會更合理一些。
UDP編程的服務器端一般步驟是:
1、創建一個socket,用函數socket();
2、設置socket屬性,用函數setsockopt();* 可選
3、綁定IP地址、端口等信息到socket上,用函數bind();
4、循環接收數據,用函數recvfrom();
5、關閉網絡連接;
UDP編程的客戶端一般步驟是:
1、創建一個socket,用函數socket();
2、設置socket屬性,用函數setsockopt();* 可選
3、綁定IP地址、端口等信息到socket上,用函數bind();* 可選
4、設置對方的IP地址和端口等屬性;
5、發送數據,用函數sendto();
6、關閉網絡連接;
服務器端程序:
#include
#include
#include
#include
#include
#include
#include
#include
#define N 256
int main(int argc, char *argv[])
{
// 0定義變量
int serverfd;
int nbytes; //獲取接收到數據字節數
struct sockaddr_in serveraddr, clientaddr;
char buf[N];
int addrlen = sizeof(struct sockaddr_in);
if(argc < 3){
puts("server
return -1;
}
// 1創建UDP套接字--socket
serverfd = socket(AF_INET, SOCK_DGRAM, 0);
if(serverfd < 0){
perror("socket err");
return -1;
}
// 2定義套接字地址--sockaddr_in
bzero(&serveraddr, addrlen);
serveraddr.sin_family = AF_INET;
serveraddr.sin_port = htons(atoi(argv[2]));
serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
// 3綁定套接字--bind
if(bind(serverfd, (struct sockaddr*)&serveraddr, addrlen) < 0){
perror("bind err");
return -1;
}
while (1) {
// 4接受來自客戶端消息 recvfrom
recvfrom(serverfd, buf, N, 0, (struct sockaddr*)&clientaddr, &addrlen);
printf("recv data = %s\n", buf);
// 5回發給客戶端 sendto
sendto(serverfd, "hello", 6, 0, (struct sockaddr*)&clientaddr, addrlen);
}
// 6關閉套接字--close
close(serverfd);
}
客戶端程序:
#include
#include
#include
#include
#include
#include
#include
#include
#define N 256
int main(int argc, char *argv[])
{
// 0定義變量
int clientfd;
struct sockaddr_in serveraddr;
char buf[N] = {0};
int addrlen = sizeof(struct sockaddr_in);
if(argc < 3){
puts("client
return -1;
}
// 1創建UDP套接字--socket
clientfd = socket(AF_INET, SOCK_DGRAM, 0);
if(clientfd < 0){
perror("socket err");
return -1;
}
// 2指定服務器地址--sockaddr_in
bzero(&serveraddr, addrlen);
serveraddr.sin_family = AF_INET;
serveraddr.sin_port = htons(atoi(argv[2]));
serveraddr.sin_addr.s_addr = inet_addr(argv[1]);
while (1) {
// 3發送消息給服務器 sendto
gets(buf);
sendto(clientfd, buf, N, 0, (struct sockaddr*)&serveraddr, addrlen);
// 4接收服務器的消息 recvfrom
recvfrom(clientfd, buf, N, 0, (struct sockaddr*)&serveraddr, &addrlen);
printf("%s\n", buf);
}
// 5關閉套接字--close
close(clientfd);
}