day2tcp和udp

1.客户端上传数据

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <head.h>

#define IP "192.168.6.160"

/**************************下载*******************************/
int tftp_download(int cfd, struct sockaddr_in sin)
{
    char buf[516] = "";

    //输入文件名
    char name[128] = "";
    printf("请输入要下载的文件名:");
    scanf("%s",name);
    getchar();

    //读写请求
    char *ptr = buf;
    unsigned short *p1 = (unsigned short*)buf;
    *p1 = htons(1);
    char *p2 = buf + 2;
    strcpy(p2,name);
    char *p3 = p2 + strlen(p2) + 1;
    strcpy(p3,"octet");

    unsigned short Ack[2];          //ACK



    //申请读写请求
    if(sendto(cfd, buf, 2+strlen(p2)+2+strlen(p3), 0, (struct sockaddr*)&sin, sizeof(sin)) < 0)
    {
        Err("sendto");
        return -1;
    }

    //创建并打开文件
    int fd = open(name,O_WRONLY|O_CREAT|O_TRUNC,0664);
    if(fd < 0)
    {
        Err("open");
    }

    //存储临时端口
    struct sockaddr_in rcvaddr;                                                                                                                 
    socklen_t addrlen = sizeof(rcvaddr);


    while(1)
    {
        //接收数据包
        bzero(buf,sizeof(buf));
        ssize_t res = recvfrom(cfd, buf, sizeof(buf), 0, (struct sockaddr*)&rcvaddr, &addrlen);
        if(res < 0)
        {
            Err("recvfrom");
            return -1;
        }
        char *q = buf + 2;
        unsigned short *k = (unsigned short*)q;

        write(fd, buf+4, res-4);

        //发送ACK包
        Ack[0]=htons(4);
        Ack[1]=htons(ntohs(*k));


        if(sendto(cfd, Ack, sizeof(Ack), 0, (struct sockaddr*)&rcvaddr, sizeof(rcvaddr)) < 0)
        {
            Err("sendto");
            return -1;
        }

        bzero(Ack,sizeof(Ack));

        //判断数据传输是否完毕
        if(res-4 < 512)
        {
            break;
        }

    }
    printf("下载完成\n");
    close(fd);

    return 0;                                                                                                                             
}


/*************************上传******************************/
int tftp_Upload(int cfd, struct sockaddr_in sin)
{
    char buf[516] = "";


    //输入文件名
    char name[128] = "";
    printf("请输入要上传的文件名:");
    scanf("%s",name);
    getchar();

    //读写请求
    char *ptr = buf;
    unsigned short *p1 = (unsigned short*)buf;
    *p1 = htons(2);
    char *p2 = buf + 2;
    strcpy(p2,name);
    char *p3 = p2 + strlen(p2) + 1;
    strcpy(p3,"octet");

    unsigned short Ack[2];          //ACK
    unsigned short operate,block;   //操作码,块编号
    ssize_t res = 0;


    //申请读写请求
    if(sendto(cfd, buf, 2+strlen(p2)+2+strlen(p3), 0, (struct sockaddr*)&sin, sizeof(sin)) < 0)
    {
        Err("sendto");
        return -1;
    }

    //打开文件
    int fd = open(name,O_RDONLY);
    if(fd < 0)
    {
        Err("open");
        return -1;                                                                                                                         
    }
     //存储临时端口
    struct sockaddr_in rcvaddr;
    socklen_t addrlen = sizeof(rcvaddr);


    bzero(Ack, sizeof(Ack));
    if ((res = recvfrom(cfd, Ack, sizeof(Ack), 0, (struct sockaddr*)&rcvaddr, &addrlen)) < 0)
    {
        Err("recvfrom");
        return -1;
    }

    // 判断服务发来的ACK包是否合法
    operate = ntohs(Ack[0]);
    block   = ntohs(Ack[1]);
    if (operate!= 4 || block != 0)
    {
        return -1;
    }
    printf("服务器ACK首次应答成功,开始传输数据……\n");


    while(1)
    {
        //发送数据包
        bzero(buf,sizeof(buf));

        res = read(fd,buf+4,512);
        if(res == 0)
        {
            printf("上传成功\n");
            break;
        }
        else if(res == -1)
        {
            Err("read");
            return -1;
        }                                                                                                                             

        unsigned short *p_operate = (unsigned short*)ptr;
        *p_operate = htons(3);

        block++;
        unsigned short *p_block = p_operate + 1;
        *p_block = htons(block);

        if(sendto(cfd, buf, 4+res, 0, (struct sockaddr*)&rcvaddr, sizeof(rcvaddr)) < 0)
        {
            Err("sendto");
            return -1;
        }


        //接收Ack包
        bzero(Ack,sizeof(Ack));
        res = recvfrom(cfd, Ack, sizeof(Ack), 0, (struct sockaddr*)&rcvaddr, &addrlen);
        if(res < 0)
        {
            Err("recvfrom");
            return -1;
        }

        operate = ntohs(Ack[0]);

        //判断数据传输是否完毕
        if(operate==4)
        {
            if(block!=ntohs(Ack[1]))
                break;
        }
        else if(operate==5)
            break;
    }
    close(fd);

    return 0;
}


                                                                                                                                          


int main(int argc, const char *argv[])
{
    //创建报式套接字
    int cfd = socket(AF_INET, SOCK_DGRAM, 0);
    if(cfd < 0)
    {
        Err("socket");
        return -1;
    }
    printf("cfd = %d\n", cfd);

    //填充服务器自身的地址信息结构体,真实的地址信息结构体根据地址族指定
    //AF_INET : man 7 ip
    struct sockaddr_in sin;
    sin.sin_family      = AF_INET;
    sin.sin_port        = htons(69);
    sin.sin_addr.s_addr = inet_addr(IP);

    int t = 0;
    while(1)
    {
        printf("******请选择功能******\n");
        printf("*******1.下载*********\n");
        printf("*******2.上传*********\n");
        printf("*******3.退出*********\n\n");
        printf("请输入:");
        scanf("%d",&t);

        switch(t)
        {
            case 1: tftp_download(cfd,sin); break;
            case 2: tftp_Upload(cfd,sin);   break;
            case 3: close(cfd);             return 0;
            default: printf("error\n");
        }
                                                                                                                           

2.思维导图

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值