C语言实现简单UDP协议服务端

一、源代码

#include
#include
#include
#include
#include
//#include
#include

static void usage(const char * str)
{
    printf("Usage:%s [IP] [PORT]\n",str);
}


//  ./server 127.0.0.1 8080
int main(int argc,char *argv[])
{
    if(argc != 3)
    {
        usage(argv[0]);
        return 1;
    }

    int sock = socket(AF_INET,SOCK_DGRAM,0);
    if(sock < 0)
    {
        perror("sock");
        return 2;
    }

    struct sockaddr_in serv_addr;
    serv_addr.sin_family = AF_INET;
    serv_addr.sin_port = htons( atoi(argv[2]) );
    serv_addr.sin_addr.s_addr = inet_addr(argv[1]);

    if( (bind(sock,(struct sockaddr*)&serv_addr,sizeof(serv_addr)) ) < 0  )
    {
        perror("bind");
        return 3;
    }


    char buf[BUFSIZ]  ;
    struct sockaddr_in clie_addr;
    socklen_t len = sizeof(clie_addr);

    while(1)
    {
        ssize_t s = recvfrom(sock,buf,sizeof(buf)-1,0,(struct sockaddr*)&clie_addr,&len );//最后一个参数是输入输出参数,
        if(s > 0)
        {
            buf[s] = 0;  //最后加上一个\0
            printf("%s:%d say # %s\n",inet_ntoa(clie_addr.sin_addr),ntohs(clie_addr.sin_port),buf);
            sendto(sock,buf,strlen(buf),0,(struct sockaddr*)&clie_addr,sizeof(clie_addr));
        }
    }

    close(sock);


    return 0;
}

二、为UDP在应用层增强可靠性的策略
关键在于两点,从应用层角度考虑:
1 提供超时重传,能避免数据报丢失。
2 提供确认序列号,可以对数据报进行确认和排序。

本端:首先在UDP数据报定义一个首部,首部包含确认序列号和时间戳,时间戳是用来计算RTT(数据报传输的往返时间),从何计算出合适的RTO(重传的超时时间)。然后以等-停的方式发送数据报,即收到对端的确认之后才发送下一个的数据报。当时间超时,本端重传数据报,同时RTO扩大为原来的两倍,重新开始计时。

对端:接受到一个数据报之后取下该数据报首部的时间戳和确认序列号,并添加本端的确认数据报首部之后发送给对端。根据此序列号对已收到的数据报进行排序并丢弃重复的数据报。

你可能感兴趣的:(编程脚印,c语言,udp)