Linux下使用Socket实现http文件下载

//test.cpp
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <sys/socket.h>
#include <arpa/inet.h>
#include <fcntl.h>
#include <unistd.h>
#include <netdb.h>

#define TARGET_URL "http://seopic.699pic.com/photo/50010/8515.jpg_wh1200.jpg"
#define TARGET_HOST "seopic.699pic.com"
#define TARGET_PORT 80 //the default port 80

static void GetIPfromDNS(char* ip_addr);
static void get_resp_header(const char *response, int *status_code, char*content_type, long* content_length);

int main(){
    int client_socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (client_socket < 0) {
        printf("invalid socket : %d\n", client_socket);
        return 0;
    }
    struct sockaddr_in addr;
    char ip_addr[64];
    memset(&addr, 0, sizeof(addr));
    GetIPfromDNS(ip_addr);
    addr.sin_family = AF_INET;
    addr.sin_addr.s_addr = inet_addr(ip_addr);
    addr.sin_port = htons(TARGET_PORT);
    int res = 0;
    res = connect(client_socket, (struct sockaddr *) &addr, sizeof(addr));
    if (res == -1){
        printf("connect failed : %d\n", res);
        return 0;
    }
    char sendbuf[1024] = {0};  
    char recvbuf[1024] = {0}; 
    int index = 0;
    char response;
    bool isGetContent = false;
    sprintf(sendbuf, \
        "GET %s HTTP/1.1\r\n"\
        "User-Agent: Mozilla/4.0 (compatible; MSIE 5.00; Windows 98)\r\n"\
        "Accept: */*\r\n"\
        "Host:%s\r\n"\
        "\r\n"\
        , TARGET_URL, TARGET_HOST);
    send(client_socket, sendbuf, strlen(sendbuf),0);
    while(recv(client_socket, &response, sizeof(response),0)!=0){
        recvbuf[index++] = response;
        if(response == '\r'){
            if(recv(client_socket, &response, sizeof(response),0)!=0){
                recvbuf[index++] = response;
                if(response == '\n'){
                       if(recv(client_socket, &response, sizeof(response),0)!=0){
                            recvbuf[index++] = response;
                            if(response == '\r'){
                                if(recv(client_socket, &response, sizeof(response),0)!=0){
                                    recvbuf[index++] = response;
                                    if(response == '\n'){
                                        isGetContent = true;
                                        printf("\n\nSUCCESS GET HEAD\n\n");
                                        break;
                                    }
                                }
                            }
                        }
                }
            }
        }
    }
    if(isGetContent == true){
        printf("---------------\n");
        printf("#### %ld ####\n\n", strlen(recvbuf));
        printf("%s", recvbuf);
        printf("---------------\n\n\n");
        fflush(stdout);
        int status_code;
        char content_type[1024]; 
        long content_length;
        get_resp_header(recvbuf, &status_code, content_type, &content_length);
        printf("%d, %s, %ld\n",  status_code, content_type, content_length);
        printf("Start write file to local disk ... .... \n");
        fflush(stdout);
        int fd = open("mydownload.jpg", O_CREAT | O_WRONLY, S_IRWXG | S_IRWXO | S_IRWXU);
        unsigned char buf[1024];
        int len = 0;
        int writeLength = 0;
        while((len = recv(client_socket, buf, 1024, 0))!=0){
            write(fd, buf, len);
            writeLength += len;
            if(writeLength == content_length){
                break;
            }
        }
        printf("\n\nTHE END\n\n");
        close(fd);
    }
    close(client_socket);
    return 0;
}

static void GetIPfromDNS(char* ip_addr){
    struct hostent *host = gethostbyname(TARGET_HOST);
    if (!host) {
        ip_addr = NULL;
        return;
    }
    for (int i = 0; host->h_addr_list[i]; i++){
        strcpy(ip_addr, inet_ntoa( * (struct in_addr*) host->h_addr_list[i]));
        break;
    }
}

/*
# status_code : 200, 503, .... 状态码
# content_type : image/jpeg 内容类型
# content_length : 560437 内容长度(字节)
*/
void get_resp_header(const char *response, int *status_code, char*content_type, long* content_length){
    char *pos = (char*)strstr(response, "HTTP/");
    if (pos)
        sscanf(pos, "%*s %d", status_code);

    pos = (char*)strstr(response, "Content-Type:");
    if (pos)
        sscanf(pos, "%*s %s", content_type);

    pos = (char*)strstr(response, "Content-Length:");
    if (pos)
        sscanf(pos, "%*s %ld",content_length);
}

运行结果

$ g++ test.cpp
$ ./a.out 
SUCCESS GET HEAD

---------------
#### 563 ####

HTTP/1.1 200 OK
Server: marco/1.6
Date: Fri, 18 Aug 2017 03:06:52 GMT
Content-Type: image/jpeg
Content-Length: 560437
Connection: keep-alive
X-Request-Id: 96aaeffc8c0ece839ba5495988d22dc5; 54f5daf9ca0effd3deacdcc88a5e54da
X-Source: U/304
ETag: "82e871eb8d245fef907c9e5ef8cd8809"
X-Slice-Complete-Length: 560437
Last-Modified: Thu, 06 Apr 2017 12:59:05 GMT
X-Slice-Size: 65536
Expires: Wed, 23 Aug 2017 17:02:27 GMT
Cache-Control: max-age=691200
Accept-Ranges: bytes
Age: 458936
Via: T.2424.H.1, V.mix-gd-can-008, T.141134.R.1, M.cun-gd-zhs-131

---------------


200, image/jpeg, 560437
Start write file to local disk ... .... 


THE END

主要注意两个点。
1 组织HTTP协议的应用层数据包发起请求。

这里写图片描述

2 利用服务器返回的数据格式中连续两次\r\n解析出头部信息(包含文件大小)和文件原始数据(字节流);
(例程中的51行到70行联系用了4个判断语句直接找出连续的\r\n)

这里写图片描述

补充一点
这里使用了gethostbyname系统函数向DNS服务器发起查询IP,但是帮助文档中已经说明这个函数不再推荐使用了

The gethostbyname*() and gethostbyaddr*() functions are obsolete.  
Applications should use getaddrinfo(3) and getnameinfo(3) instead.

HTTP使用了TCP作为传输层协议,所以会有3次握手的过程,标准的3次握手过程:

这里写图片描述

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 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、付费专栏及课程。

余额充值