更多网络程序设计的文章见:目录
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;
}