更多網絡程序設計的文章見:目錄
UDP特點
- 無連接
- 基於消息的數據傳輸服務
- 不可靠
- 一般情況下UDP更高效
UDP回射客戶/服務器
- echosrv.c
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define ERR_EXIT(m) \
do\
{\
perror(m);\
exit(EXIT_FAILURE);\
}while (0)
void echo_srv(int sock)
{
char recvbuf[1024] = {0};
struct sockaddr_in peeraddr;
socklen_t peerlen;
int n;
while(1)
{
peerlen = sizeof(peeraddr);
memset(recvbuf, 0, sizeof(recvbuf));
n = recvfrom(sock, recvbuf, sizeof(recvbuf), 0, (struct sockaddr*)&peeraddr, &peerlen);
if(n == -1)
{
if(errno == EINTR)
continue;
ERR_EXIT("recvbuf");
}
else if(n > 0)
{
fputs(recvbuf, stdout);
sendto(sock, recvbuf, n, 0, (struct sockaddr*)&peeraddr, peerlen);
}
}
close(sock);
}
int main(void)
{
//步驟一:創建套接口
int sock;
if((sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0)
//if((sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0)//第三個參數協議可以用0表示自動選擇協議
ERR_EXIT("socket");
//步驟二:初始化一個地址,綁定它
struct sockaddr_in servaddr;
memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(5188);
servaddr.sin_addr.s_addr = htonl(INADDR_ANY);
if(bind(sock, (struct sockaddr*)&servaddr, sizeof(servaddr)) < 0)
ERR_EXIT("bind");
//不需要監聽,下一步是回射服務器
echo_srv(sock);
return 0;
}
- echocli.c
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define ERR_EXIT(m) \
do\
{\
perror(m);\
exit(EXIT_FAILURE);\
}while (0)
void echo_cli(int sock)
{
struct sockaddr_in servaddr;
memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(5188);
servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");
//通常情況不需要綁定
char sendbuf[1024] = {0};
char recvbuf[1024] = {0};
while(fgets(sendbuf, sizeof(sendbuf), stdin) != NULL)
{
sendto(sock, sendbuf, strlen(sendbuf), 0, (struct sockaddr*)&servaddr, sizeof(servaddr));
recvfrom(sock, recvbuf, sizeof(recvbuf), 0, NULL, NULL);
fputs(recvbuf, stdout);
memset(sendbuf, 0, sizeof(sendbuf));
memset(recvbuf, 0, sizeof(recvbuf));
}
close(sock);
}
int main(void)
{
//步驟一:創建套接字
int sock;
if((sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0)//第三個參數協議可以用0表示自動選擇協議
ERR_EXIT("socket");
//步驟二:調用一個回射客戶端的函數
echo_cli(sock);
return 0;
}
UDP注意點
- UDP報文可能會丟失、重複
- UDP報文可能會亂序
- UDP缺乏流量控制
- UDP協議數據報文截斷
- recvfrom返回0,不代表連接關閉,因爲UDP是無連接的
- ICMP異步錯誤(解決方法:UDP connect)
- UDP外出接口的確定
改進的echocli.c代碼
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#define ERR_EXIT(m) \
do\
{\
perror(m);\
exit(EXIT_FAILURE);\
}while (0)
void echo_cli(int sock)
{
struct sockaddr_in servaddr;
memset(&servaddr, 0, sizeof(servaddr));
servaddr.sin_family = AF_INET;
servaddr.sin_port = htons(5188);
servaddr.sin_addr.s_addr = inet_addr("127.0.0.1");
//通常情況不需要綁定
connect(sock, (struct sockaddr*)&servaddr, sizeof(servaddr));//解決ICMP異步錯誤,但這個連接和TCP的不一樣
int ret;
char sendbuf[1024] = {0};
char recvbuf[1024] = {0};
while(fgets(sendbuf, sizeof(sendbuf), stdin) != NULL)
{
/*sendto(sock, sendbuf, strlen(sendbuf), 0, (struct sockaddr*)&servaddr, sizeof(servaddr));
sendto(sock, sendbuf, strlen(sendbuf), 0, NULL, 0);*/
send(sock, sendbuf, strlen(sendbuf), 0);
ret = recvfrom(sock, recvbuf, sizeof(recvbuf), 0, NULL, NULL);
if(ret == -1)
{
if(errno == EINTR)
continue;
ERR_EXIT("recvbuf");
}
fputs(recvbuf, stdout);
memset(sendbuf, 0, sizeof(sendbuf));
memset(recvbuf, 0, sizeof(recvbuf));
}
close(sock);
}
int main(void)
{
//步驟一:創建套接字
int sock;
if((sock = socket(PF_INET, SOCK_DGRAM, 0)) < 0)//第三個參數協議可以用0表示自動選擇協議
ERR_EXIT("socket");
//步驟二:調用一個回射客戶端的函數
echo_cli(sock);
return 0;
}