带有超时的udp客户端

#include <stdio.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <errno.h>
#include <string.h>
#include <signal.h>
#include <unistd.h>
static int sig_alrm(int);
const int maxline=4096;
int readable_timeo(int fd, int sec) 
{
    fd_set rset;
    struct timeval tv;
    FD_ZERO(&rset);
    FD_SET(fd, &rset);
    tv.tv_sec=sec;
    tv.tv_usec=0;
    return select(fd+1, &rset, NULL, NULL, &tv);
}
void dg_cli(FILE *fp, int sockfd, const struct sockaddr * pservaddr, socklen_t servlen) 
{
    int n;
    char sendline[maxline], recvline[maxline+1];
    char errbuff[maxline];
    while(fgets(sendline, maxline, fp) != NULL) 
    {
        if(sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen)<0) 
        {
            strerror_r(errno, errbuff, maxline);
            fprintf(stderr, "sendto error: %sn", errbuff);
            exit(-1);
        }
        if((n=recvfrom(sockfd, recvline, maxline, 0, NULL, NULL))<0) {
            strerror_r(errno, errbuff, maxline);
            fprintf(stderr, "recvfrom error: %sn", errbuff);
            exit(-1);
        }
        recvline[n]=0;
        fputs(recvline, stdout);
    }
}
void dg_cli_so(FILE *fp, int sockfd, const struct sockaddr * pservaddr, socklen_t servlen) {
    int n;
    char sendline[maxline], recvline[maxline+1];
    char errbuff[maxline];
    struct timeval tv;
    tv.tv_sec=5;
    tv.tv_usec=0;
    if(setsockopt(sockfd, SOL_SOCKET, SO_RCVTIMEO, &tv, sizeof(tv))==-1) {
        strerror_r(errno, errbuff, maxline);
        fprintf(stderr, "setsockopt error: %sn", errbuff);
        exit(-1);
    }
    while(fgets(sendline, maxline, fp) != NULL) {
        if(sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen)<0) {
            strerror_r(errno, errbuff, maxline);
            fprintf(stderr, "sendto error: %sn", errbuff);
            exit(-1);
        }
        n=recvfrom(sockfd, recvline, maxline, 0, NULL, NULL);
        if(n<0) {
            if(errno==EWOULDBLOCK) {
                fprintf(stderr, "socket timeoutn");
                continue;
            } else {
                strerror_r(errno, errbuff, maxline);
                fprintf(stderr, "recvfrom error: %sn", errbuff);
                exit(-1);
            }
        }
        recvline[n]=0;
        fputs(recvline, stdout);
    }
}
void dg_cli_select(FILE *fp, int sockfd, const struct sockaddr * pservaddr, socklen_t servlen) {
    int n;
    char sendline[maxline], recvline[maxline+1];
    char errbuff[maxline];
    while(fgets(sendline, maxline, fp) != NULL) {
        if(sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen)<0) {
            strerror_r(errno, errbuff, maxline);
            fprintf(stderr, "sendto error: %sn", errbuff);
            exit(-1);
        }
        if(readable_timeo(sockfd, 5)==0) {
            fprintf(stderr, "socket timeoutn");
        } else {
            if((n=recvfrom(sockfd, recvline, maxline, 0, NULL, NULL))<0) {
                strerror_r(errno, errbuff, maxline);
                fprintf(stderr, "recvfrom error: %sn", errbuff);
                exit(-1);
            }
            recvline[n]=0;
            fputs(recvline, stdout);
        }
    }
}
void dg_cli_alrm(FILE *fp, int sockfd, const struct sockaddr * pservaddr, socklen_t servlen) {
    int n;
    char sendline[maxline], recvline[maxline+1];
    char errbuff[maxline];
    if(signal(SIGALRM, sig_alrm)<0) {
        strerror_r(errno, errbuff, maxline);
        fprintf(stderr, "signal error: %sn", errbuff);
        exit(-1);
    }
    while(fgets(sendline, maxline, fp) != NULL) {
        if(sendto(sockfd, sendline, strlen(sendline), 0, pservaddr, servlen)<0) {
            strerror_r(errno, errbuff, maxline);
            fprintf(stderr, "sendto error: %sn", errbuff);
            exit(-1);
        }
        alarm(5);
        if((n=recvfrom(sockfd, recvline, maxline, 0, NULL, NULL))<0) {
            if(errno==EINTR) {
                fprintf(stderr, "socket timeoutn");
            } else {
                strerror_r(errno, errbuff, maxline);
                fprintf(stderr, "recvfrom error: %sn", errbuff);
                exit(-1);
            }
        } else {
            alarm(0);
            recvline[n]=0;
            fputs(recvline, stdout);
        }
    }
}
int main(int argc, char **argv) {
    int sockfd, n;
    char recvline[maxline+1], errbuff[maxline];
    struct sockaddr_in servaddr;
    if(argc != 2) {
        fprintf(stderr, "usage: a.out <IPaddress>n");
        exit(-1);
    }
    if((sockfd=socket(AF_INET, SOCK_DGRAM, 0))<0) {
        strerror_r(errno, errbuff, maxline);
        fprintf(stderr, "socket error: %sn", errbuff);
        exit(-1);
    }
    memset(&servaddr, 0, sizeof(servaddr));
    servaddr.sin_family=AF_INET;
    servaddr.sin_port=htons(9999);
    if(inet_pton(AF_INET, argv[1], &servaddr.sin_addr)<=0) {
        fprintf(stderr, "inet_pton error for %s", argv[1]);
        exit(-1);
    }
    dg_cli_so(stdin, sockfd, (struct sockaddr *)&servaddr, sizeof(servaddr));
    return 0;
}
static int sig_alrm(int signo) {
    printf("signal alarmn");
    exit -1;
}

  

posted @ 2022-09-30 12:55  东宫得臣  阅读(21)  评论(0编辑  收藏  举报