UNIX网络编程卷1 回射客户程序 UDP 超时设置
来源:互联网 发布:博知教育 编辑:程序博客网 时间:2024/05/16 14:56
本文为senlie原创,转载请保留此地址:http://blog.csdn.net/zhengsenlie
最初代码:
#include"unp.h"intmain(int argc, char **argv){intsockfd;struct sockaddr_inservaddr;if (argc != 2)err_quit("usage: udpcli <IPaddress>"); //1.指明服务器的 IP 地址和端口bzero(&servaddr, sizeof(servaddr));servaddr.sin_family = AF_INET;servaddr.sin_port = htons(SERV_PORT);Inet_pton(AF_INET, argv[1], &servaddr.sin_addr); //2.创建一个 UDP 套接字sockfd = Socket(AF_INET, SOCK_DGRAM, 0); //3.把 UDP 套接字和服务器套接字地址结构传递给 dg_cli 函数dg_cli(stdin, sockfd, (SA *) &servaddr, sizeof(servaddr));exit(0);}#include"unp.h"voiddg_cli(FILE *fp, int sockfd, const SA *pservaddr, socklen_t servlen){intn;charsendline[MAXLINE], recvline[MAXLINE + 1]; //1.fgets 从标准输入读入一个文本行while (Fgets(sendline, MAXLINE, fp) != NULL) { //2.用 sendto 从sockfd 套接字将文本行发送给pservaddr指明的服务器Sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen); //3.用 recvfrom 从sockfd 套接字读取服务器的回射n = Recvfrom(sockfd, recvline, MAXLINE, 0, NULL, NULL); //4.用 fputs 将回射文本显示到标准输出recvline[n] = 0;/* null terminate */Fputs(recvline, stdout);}}
问题: recvfrom 指定的第五和第六个参数是空指,这样任何进程不论是在与本客户进程相同的主机上
还是在不同的主机上,都可以向本客户的 IP 地址和端口发送数据报
改善1:验证接收到的响应,保留来自数据报所发往服务器的应答,而忽略任何其他数据报。
/** * UDP 验证服务器地址 * **/#include"unp.h"voiddg_cli(FILE *fp, int sockfd, const SA *pservaddr, socklen_t servlen){intn;charsendline[MAXLINE], recvline[MAXLINE + 1];socklen_tlen;struct sockaddr*preply_addr; //1.分配另一个套接字地址结构preply_addr = Malloc(servlen);while (Fgets(sendline, MAXLINE, fp) != NULL) {Sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen);len = servlen;n = Recvfrom(sockfd, recvline, MAXLINE, 0, preply_addr, &len); //2.比较返回的地址if (len != servlen || memcmp(pservaddr, preply_addr, len) != 0) {printf("reply from %s (ignored)\n",Sock_ntop(preply_addr, len));continue;}recvline[n] = 0;/* null terminate */Fputs(recvline, stdout);}}
改善1问题:如果没有接收到服务器返回的应答就一直阻塞在 recvfrom
改善2.1:使用 SIGALRM 为 recvfrom 设置超时
#include"unp.h"static voidsig_alrm(int);voiddg_cli(FILE *fp, int sockfd, const SA *pservaddr, socklen_t servlen){intn;charsendline[MAXLINE], recvline[MAXLINE + 1];//1.设置超时信号 SIGALRM 的处理函数Signal(SIGALRM, sig_alrm);while (Fgets(sendline, MAXLINE, fp) != NULL) {Sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen);//2.每次调用 recvfrom 前通过调用 alarm 设置一个 5 秒钟的超时alarm(5);if ( (n = recvfrom(sockfd, recvline, MAXLINE, 0, NULL, NULL)) < 0) {//3.如果 errno 为 EINTR,即 recvfrom 因超时而中断了if (errno == EINTR) fprintf(stderr, "socket timeout\n");elseerr_sys("recvfrom error");} //4.如果读到一行来自服务器的文本,那就关掉报警时钟并输出服务器的应答else {alarm(0);recvline[n] = 0;/* null terminate */Fputs(recvline, stdout);}}}//SIGALRM 信号处理函数static voidsig_alrm(int signo){return;/* just interrupt the recvfrom() */}
改善2.2:使用 select 为 recvfrom 设置超时
/* include readable_timeo */#include"unp.h"//等待一个描述符最多在指定的秒数内变为可读intreadable_timeo(int fd, int sec){fd_setrset;struct timevaltv;//1.设置描述符集FD_ZERO(&rset);FD_SET(fd, &rset);//2.设置等待的秒数tv.tv_sec = sec;tv.tv_usec = 0;//3.阻塞在select 上return(select(fd+1, &rset, NULL, NULL, &tv));/* 4> 0 if descriptor is readable */}/* end readable_timeo */intReadable_timeo(int fd, int sec){intn;if ( (n = readable_timeo(fd, sec)) < 0)err_sys("readable_timeo error");return(n);}#include"unp.h"voiddg_cli(FILE *fp, int sockfd, const SA *pservaddr, socklen_t servlen){intn;charsendline[MAXLINE], recvline[MAXLINE + 1];while (Fgets(sendline, MAXLINE, fp) != NULL) {Sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen);//1.直到 readable_timeo 告诉所关注的描述符已变为可读后我们才调用 recvfrom ,这一点保证 recvfrom 不会阻塞if (Readable_timeo(sockfd, 5) == 0) {fprintf(stderr, "socket timeout\n");} else {//2.接收数据并输出到 stdoutn = Recvfrom(sockfd, recvline, MAXLINE, 0, NULL, NULL);recvline[n] = 0;/* null terminate */Fputs(recvline, stdout);}}}
改善2.3:使用 SO_RCVTIMEO 套接字选项为 recvfrom 设置超时
#include"unp.h"voiddg_cli(FILE *fp, int sockfd, const SA *pservaddr, socklen_t servlen){intn;charsendline[MAXLINE], recvline[MAXLINE + 1];struct timevaltv;//1.在套接字上设置超时tv.tv_sec = 5;tv.tv_usec = 0;Setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv));while (Fgets(sendline, MAXLINE, fp) != NULL) {Sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen);2. 接收数据,如果超时的话,recvfrom 会返回 EWOULDBLOCK 错误n = recvfrom(sockfd, recvline, MAXLINE, 0, NULL, NULL);if (n < 0) {if (errno == EWOULDBLOCK) {fprintf(stderr, "socket timeout\n");continue;} elseerr_sys("recvfrom error");}recvline[n] = 0;/* null terminate */Fputs(recvline, stdout);}}
0 0
- UNIX网络编程卷1 回射客户程序 UDP 超时设置
- UNIX网络编程卷1 回射客户程序 TCP客户程序设计范式
- Unix网络编程 卷1 第8章:基本UDP套接字编程(UDP回射客户/服务器程序: main/dg_echo/dg_cli)
- 学习 UNIX网络编程卷1:套接字 笔记1-实现一个简单的回射客户服务器程序
- 《UNIX网络编程 卷1》 笔记: UDP应用实现超时重传机制
- 【Unix 网络编程】UDP 客户/服务器简单 Socket 程序
- 《UNIX网络编程 卷1》 笔记: TCP 客户/服务器程序示例
- 《UNIX网络编程 卷1》 笔记: 非阻塞式connect—web客户程序
- 《UNIX网络编程 卷1》 笔记: 多线程—web客户程序
- UNIX网络编程卷1:套接字联网-第5章:TCP客户/服务器程序示例
- 简单的UDP回射服务程序与客户程序(修改自Unix网络编程一书)
- UNIX网络编程卷1 时间获取程序客户端 UDP 协议无关
- UNIX网络编程卷1 时间获取程序服务器 UDP 协议无关
- Unix网络编程 第一卷 套接口API 第五章 TCP客户/服务器程序例子
- UNIX网络编程卷一 笔记 第五章 TCP客户/服务器程序示例
- UNIX网络编程卷一:第五章 TCP客户/服务器程序实例
- UNIX网络编程卷一 第五章 TCP客户/服务器程序示例
- UNIX网络编程卷1 回射服务器程序 TCP服务器程序设计范式 四个版本
- office2010中的word如何隐藏回车换行符??
- Python: Why Use Built-in Types?
- ubuntu使用
- 【leetcode】merge-sorted-array
- 解决unity3d 多平台下完美本地文本文件读取的方法
- UNIX网络编程卷1 回射客户程序 UDP 超时设置
- HDU 3292 No more tricks, Mr Nanguo(佩尔方程,矩阵快速幂)
- poj 2887 Big String
- unix/linux编程实践教程--more命令
- POJ 3463 Sightseeing Dijkstra最短路&最短路-1的路径数(计数)
- makefile write_janseman99_1
- 看看别人家的孩子
- User has no SELECT privilege on V$SESSION(执行计划相关)
- python2 python3语法区别