Linux C用http协议下载文件

#include <stdio.h>
#include <stdlib.h>
#include <limits.h>
#include <sys/time.h>
#include <sys/select.h>
#include <netdb.h>
#include <ctype.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <unistd.h>
#include <netinet/in.h>
#define RECV_BUF_LEN (1024*100)
struct hostent *host;
char domain[256] = {0};
int port = 0;
int ret_status(char *recv_buf)
{
    if(NULL == recv_buf)
    {   
        printf("recv_buf is NULL\n");
        return -1;
    }
    int http_status = 0;
    char line[256] = {0} ;
    char *rest ;
    rest = strstr(recv_buf,"\r\n");
    if ( rest != NULL)
    {
        memcpy(line,recv_buf,rest-recv_buf);
        if(strstr(line,"200"))
        {
            http_status = 200;
        }
        else if(strstr(line,"302") || strstr(line,"301"))
        {
            http_status = 302;
        }
        else
        {
            http_status = -1;
        }
    }
    return http_status;
}
char *afresh_url(char *recv_buf)
{
    char *rest = NULL;
    char *new_url = NULL;
    char *line = NULL;
    if(NULL == recv_buf)
    {
        return NULL;
    }
    new_url = (char *)malloc(RECV_BUF_LEN);
    if(NULL == new_url)
    {
        printf("malloc new_url error!\n");
        return NULL;
    }
    if( strstr(recv_buf,"Location:") == NULL)
    {
        return NULL;
    }
    else
    {
        rest = strstr(recv_buf,"Location:")+strlen("location: ");
        line = strstr(rest,"\r\n");
        memcpy(new_url, rest,line-rest);
    }
    return new_url;
}

long ret_file_size(char *recv_buf)
{
    long file_size = 0;
    char *rest = NULL;
    char *line = NULL;
    char actual_size[RECV_BUF_LEN] = {0} ;
   
    if( NULL == recv_buf)
    {
        printf("recv %s is NULL\n",recv_buf);
        return -1;
    }
    if((strstr(recv_buf,"Content-Length")) == NULL)
    {
//        printf("Content-Length is NULL\n");
        return -1;
    }
    rest = strstr(recv_buf,"Content-Length:")+strlen("Content-Length: ");
    line = strstr(rest,"\r\n");
    memcpy(actual_size,rest,line-rest);
    file_size = atoi(actual_size);
    return file_size;
}

char *down_local_file(char *src)
{
        if(NULL == src)
        {
                return NULL;
        }
        char *line = NULL;
        char file[100] = {0};
        char *content = NULL;
        char *file_name;
        int len = 0;

        if(strstr(src,"Content-Disposition") == 0)
        {
                return NULL;
        }
        else
        {
                file_name = (char *)malloc(100);
                if(NULL == file_name)
                {
                        return NULL;
                }
                line = strstr(src,"filename=") + 9;
                if( NULL != line)
                {
                        content = strstr(line,"\r\n");
                        if( content != NULL)
                        {
                                len =  strlen(line) - strlen(content);
                                memcpy(file_name,line,len);
                                line[len] = 0;
                        }
                }
        }
        return file_name;
}

char *down_file(char *src, char c)
{
    if(NULL == src)
    {
        printf("The src %s is null\n",src);
        exit(-1);
    }
    int len = 0;
    len = strlen(src);
    while(src[len-1])
    {
        if( strchr(src + (len - 1),c))
        {
            return (src + (len - 1));
        }
        else
        {
            len--;
        }
    }
    return NULL;
}

char *http_request(char *url)
{
    char *url_dress = NULL;
    char *file_dress = NULL;
    char *ret = NULL;
    char *request;
    struct hostent *host;
   
    url_dress = url+strlen("http://");
    file_dress = strchr(url_dress,'/');
    if (file_dress)
    {
        memcpy(domain,url_dress,file_dress-url_dress);
    }
    ret = strstr(domain,":");
    if(ret)
    {
        port = atoi(ret);
    }
    else
    {
        port = 80;
    }
    request = (char *)malloc(RECV_BUF_LEN);
    if(NULL == request)
    {
        printf("malloc request error!\n");
        exit(-1);
    }
    sprintf(request,"GET %s HTTP/1.1\r\nHost:%s\r\n\
User-Agent:Mozilla/5.0 (Windows NT 6.1; rv:12.0) Gecko/20100101 Firefox/12.0\r\n\
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8\r\n\
Accept-Language: zh-cn,zh;q=0.8,en-us;q=0.5,en;q=0.3\r\n\
Accept-Encoding: gzip,deflate\r\n\
Connection: keep-alive\r\n\r\n",file_dress,domain);
    return request;
}

int main(int argc, char *argv[])
{
    char *url;
    int sockefd;
    char *ret = NULL;
    char *buffer ;
    char *head_buffer = NULL;
    char header[RECV_BUF_LEN] = {0};
    struct sockaddr_in server_addr;
    FILE *fp = NULL;
    char *request;
    int i = 0;
    char *file_dress = NULL;
    char *file_name;
    char *local_file = NULL;
    int send = 0;
    int totalsend = 0;
    int nbytes = 0;
    char *url_address = NULL;
    int http_status = 0;
    struct timeval start_time,end_time;
    float timeuse;
    FILE *file ;
    char url_link[200] = {0};
    char *url_ret;
    if (argc != 2)
    {
        fprintf(stderr,"Usage: %s web address \n",argv[0]);
        exit(-1);
    }
    url = argv[1];
    do {
        request = http_request(url);
        if((host = gethostbyname(domain)) == NULL)
        {
            printf("Get host name error!\n");
            exit(-1);
        }
        gettimeofday(&start_time,NULL);
        if((sockefd = socket(AF_INET,SOCK_STREAM,0)) == -1)
        {
            printf("Socket error!\n");
            exit(-1);
        }
        bzero(&server_addr,sizeof(server_addr));
        server_addr.sin_family = AF_INET;
        server_addr.sin_port = htons(port);
        server_addr.sin_addr = *((struct in_addr *)host->h_addr);

        if (connect(sockefd, (struct sockaddr *)(&server_addr),sizeof(struct sockaddr)) == -1)
        {
            printf("connect error!\n");
            exit(-1);
        }
        send = 0;
        totalsend = 0;
        nbytes = strlen(request);
        while(totalsend < nbytes)
        {
            send = write(sockefd ,request + totalsend,nbytes - totalsend);
            if (send == -1)
            {
                printf("send error!\n");
                exit(-1);
            }
            totalsend += nbytes;
            printf("%d bytes send OK!\n",totalsend);
        }

        long file_size = 0;
        int mark = 1;
        int count=0;
        buffer = (char *)malloc(RECV_BUF_LEN);
        do   
        {
            memset(buffer,0,RECV_BUF_LEN);
            nbytes = recv(sockefd,buffer,RECV_BUF_LEN,0);
            if (mark > 0) {
                mark = 0;
                head_buffer = strstr(buffer,"\r\n\r\n")+4;
                memcpy(header,buffer,head_buffer-buffer);
                printf("%s\n",header);
                file_size = ret_file_size(header);
                http_status = ret_status(header);
                if(http_status == 200)
                {
                    file_name = down_local_file(header);
                    if(NULL == file_name)
                    {
                        file_name = down_file(url,'/')+1;
                        fp = fopen(file_name,"w");
                        if(NULL == fp)
                        {
                            printf("open file %s failed!",local_file);
                            exit(-1);
                        }
                    }
                    else
                    {
                        fp = fopen(file_name,"w");
                        if(NULL == fp)
                        {
                            printf("open file %s failed!",local_file);
                            exit(-1);
                        }
                    }
                    count = fwrite(head_buffer,1,nbytes-(head_buffer - buffer),fp);
        //        printf("%s\t%d",head_buffer,strlen(head_buffer));
                //printf("head_buffer   %s\n",head_buffer);
                //printf("--------------\n");
                    url = NULL;
                }
                else if(http_status == 302)
                {
                    url = afresh_url(header);
                }
            }
            else
            {
                fwrite(buffer,1,nbytes,fp);
                fflush(fp);
                count+=nbytes;
            }
            if(count == file_size)
            {
                break;
            }
        }while(nbytes > 0);
        printf("total:%d\n",count);
        free(request);
        free(buffer);
        close(sockefd);
    }while(url != NULL );
    gettimeofday(&end_time,NULL);
    timeuse = 1000000*(end_time.tv_sec - start_time.tv_sec) + end_time.tv_usec - start_time.tv_usec;
    timeuse /= 1000000;
    printf("Used time: %f\n",timeuse);
    fclose(fp);
    free(url);
    return 0;
}

转载于:https://www.cnblogs.com/emodj/archive/2013/03/27/2985239.html

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Linux下,我们可以使用C语言来实现FTP下载文件的功能。下面是一个简单的实现过程: 首先,我们需要包含相关的头文件: ```c #include <sys/socket.h> #include <arpa/inet.h> #include <unistd.h> #include <stdio.h> #include <string.h> #include <stdlib.h> ``` 然后,我们需要定义一些常量和变量: ```c #define BUFFER_SIZE 1024 #define FTP_SERVER_IP "ftp.server.com" // FTP服务器的IP地址 #define FTP_SERVER_PORT 21 // FTP服务器的端口号 #define FTP_USERNAME "username" // FTP登录用户名 #define FTP_PASSWORD "password" // FTP登录密码 #define FTP_FILE_PATH "/path/to/file.txt" // 要下载文件在FTP服务器中的路径 #define LOCAL_FILE_PATH "/path/to/save/file.txt" // 下载文件保存在本地的路径 ``` 接下来,创建一个TCP套接字并连接到FTP服务器: ```c int sockfd; struct sockaddr_in server_addr; if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) { perror("socket creation failed"); exit(EXIT_FAILURE); } memset(&server_addr, 0, sizeof(server_addr)); server_addr.sin_family = AF_INET; server_addr.sin_port = htons(FTP_SERVER_PORT); if (inet_pton(AF_INET, FTP_SERVER_IP, &server_addr.sin_addr) <= 0) { perror("invalid address"); exit(EXIT_FAILURE); } if (connect(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) < 0) { perror("connection failed"); exit(EXIT_FAILURE); } ``` 然后,发送FTP登录命令和用户名: ```c char recv_buffer[BUFFER_SIZE]; char send_buffer[BUFFER_SIZE]; memset(recv_buffer, 0, sizeof(recv_buffer)); memset(send_buffer, 0, sizeof(send_buffer)); recv(sockfd, recv_buffer, sizeof(recv_buffer), 0); sprintf(send_buffer, "USER %s\r\n", FTP_USERNAME); send(sockfd, send_buffer, strlen(send_buffer), 0); recv(sockfd, recv_buffer, sizeof(recv_buffer), 0); ``` 接着,发送FTP登录命令和密码: ```c sprintf(send_buffer, "PASS %s\r\n", FTP_PASSWORD); send(sockfd, send_buffer, strlen(send_buffer), 0); recv(sockfd, recv_buffer, sizeof(recv_buffer), 0); ``` 然后,发送FTP下载文件命令: ```c sprintf(send_buffer, "RETR %s\r\n", FTP_FILE_PATH); send(sockfd, send_buffer, strlen(send_buffer), 0); recv(sockfd, recv_buffer, sizeof(recv_buffer), 0); ``` 最后,接收并保存下载文件: ```c FILE *file = fopen(LOCAL_FILE_PATH, "wb"); if (file == NULL) { perror("file creation failed"); exit(EXIT_FAILURE); } int n; while ((n = recv(sockfd, recv_buffer, sizeof(recv_buffer), 0)) > 0) { fwrite(recv_buffer, sizeof(char), n, file); } fclose(file); ``` 最后,关闭套接字并释放资源: ```c close(sockfd); ``` 以上是简单实现FTP下载文件的C语言代码,通过这段代码,我们可以在Linux下实现FTP下载文件的功能。当然,这只是一个简单的实现示例,实际应用可能需要更完善的错误处理和其他功能。 ### 回答2: 在Linux下使用C语言实现FTP下载文件的过程如下: 首先,需要连接FTP服务器。可以使用socket函数创建一个套接字,然后使用connect函数连接到FTP服务器的IP地址和端口号(通常为21)。 接下来,需要进行FTP的用户认证。发送USER命令和PASS命令,将FTP服务器的用户名和密码发送给服务器进行认证。 认证成功后,可以发送RETR(Retrieve)命令请求下载文件。发送该命令时,需要指定要下载文件名。发送之前需要使用PASV命令进入被动模式,使得FTP服务器告知本地计算机开启哪个端口进行数据传输。 然后,使用数据连接进行文件传输。使用accept函数监听FTP服务器返回的数据连接请求,并使用accept函数接受连接。在建立连接后,使用recv函数从FTP服务器接收文件数据,并使用write函数将数据写入本地文件。 最后,下载完成后,使用QUIT命令关闭FTP连接。 需要注意的是,FTP协议本身是不安全的,所有的数据都是明文传输。如果需要安全的文件传输,可以考虑使用SFTP(SSH File Transfer Protocol)协议。 以上就是在Linux下使用C语言实现FTP下载文件的基本步骤。实际编写过程中需要注意处理错误和异常情况,并且对FTP协议的各种命令和响应进行适当的解析和处理。 ### 回答3: 在Linux下,可以使用C语言来实现FTP下载文件的功能。下面是一个简单的示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <string.h> #include <sys/socket.h> #include <netinet/in.h> #include <arpa/inet.h> #include <netdb.h> #include <errno.h> #define BUFFER_SIZE 1024 int main() { int sockfd, bytes; char buffer[BUFFER_SIZE]; struct sockaddr_in server_addr; struct hostent *host; // 创建套接字 sockfd = socket(AF_INET, SOCK_STREAM, 0); if (sockfd == -1) { perror("Error creating socket"); exit(1); } // 设置服务器地址结构 server_addr.sin_family = AF_INET; server_addr.sin_port = htons(21); // FTP默认端口 host = gethostbyname("ftp.example.com"); // FTP服务器地址 server_addr.sin_addr = *((struct in_addr *)host->h_addr); memset(&(server_addr.sin_zero), '\0', 8); // 连接到服务器 if (connect(sockfd, (struct sockaddr *)&server_addr, sizeof(struct sockaddr)) == -1) { perror("Error connecting to server"); exit(1); } // 接收服务器的欢迎信息 memset(buffer, 0, BUFFER_SIZE); bytes = recv(sockfd, buffer, BUFFER_SIZE, 0); if (bytes == -1) { perror("Error receiving welcome message"); exit(1); } printf("%s", buffer); // 发送用户名和密码 char username[] = "ftpuser"; char password[] = "ftppassword"; sprintf(buffer, "USER %s\r\n", username); send(sockfd, buffer, strlen(buffer), 0); memset(buffer, 0, BUFFER_SIZE); bytes = recv(sockfd, buffer, BUFFER_SIZE, 0); printf("%s", buffer); sprintf(buffer, "PASS %s\r\n", password); send(sockfd, buffer, strlen(buffer), 0); memset(buffer, 0, BUFFER_SIZE); bytes = recv(sockfd, buffer, BUFFER_SIZE, 0); printf("%s", buffer); // 下载文件 char filename[] = "file.txt"; // 要下载文件名 sprintf(buffer, "RETR %s\r\n", filename); send(sockfd, buffer, strlen(buffer), 0); // 创建本地文件 FILE *fp = fopen(filename, "w"); if (fp == NULL) { perror("Error creating local file"); exit(1); } // 接收数据并写入文件 while ((bytes = recv(sockfd, buffer, BUFFER_SIZE, 0)) > 0) { if (fwrite(buffer, sizeof(char), bytes, fp) != bytes) { perror("Error writing to file"); exit(1); } } // 关闭文件和套接字 fclose(fp); close(sockfd); return 0; } ``` 在该示例代码中,通过创建一个套接字并与FTP服务器建立连接。然后发送用户名和密码进行登录,然后发送RETR命令来下载指定的文件。接收到的文件数据写入本地的文件中。最后关闭文件和套接字。 需要注意的是,上述示例代码仅为简单示例,实际使用时还需要增加错误处理、异常处理、命令交互等功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值