22、socket编程实现文件传输功能

本文详细介绍了一个实用的Socket文件传输程序的实现,包括如何处理不确定大小的文件,通过循环调用send()和recv()函数进行数据传输,以及如何使用shutdown()函数发送FIN包通知文件传输完成。

这节我们来完成 socket 文件传输程序,这是一个非常实用的例子。要实现的功能为:client 从 server下载一个文件并保存到本地。

编写这个程序需要注意两个问题:
(1)文件大小不确定,有可能比缓冲区大很多,调用一次 write()/send() 函数不能完成文件内容的发送。接收数据时也会遇到同样的情况。
要解决这个问题,可以使用 while 循环,例如:

//server代码
char buf[512] = {0};
size_t count;
/* 读取发送文件 */
while ((count = fread(buf, 1, 512, file)) > 0)
{
    send(clientSock, buf, count, 0);
}

//client代码
char buf[512] = {0};
ssize_t len;
/* 循环接收数据,写入文件 */
while (0 != (len = recv(sock, buf, 512, 0)))
{
    fwrite(buf, len, 1, file);
}

对于 Server 端的代码,当读取到文件末尾,fread() 会返回 0,结束循环。

对于 Client 端代码,有一个关键的问题,就是文件传输完毕后让 recv() 返回 0,结束 while 循环。

注意:读取完缓冲区中的数据 recv() 并不会返回 0,而是被阻塞,直到缓冲区中再次有数据。

(2)Client 端如何判断文件接收完毕,也就是上面提到的问题——何时结束 while 循环。


最简单的结束 while 循环的方法当然是文件接收完毕后让 recv() 函数返回 0,那么,如何让 recv() 返回 0 呢?recv() 返回 0 的唯一时机就是收到FIN包时

FIN 包表示数据传输完毕,计算机收到 FIN 包后就知道对方不会再向自己传输数据,当调用 read()/recv() 函数时,如果缓冲区中没有数据,就会返回 0,表示读到了”socket文件的末尾“。

这里我们调用 shutdown() 来发送FIN包:server 端直接调用 close()/closesocket() 会使输出缓冲区中的数据失效,文件内容很有可能没有传输完毕连接就断开了,而调用 shutdown() 会等待输出缓冲区中的数据传输完毕。

Linux代码:

//服务端

#include <stdio.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <errno.h>

int main(int c, char * * v)
{
    /* 检查文件是否存在 */
    char *fileName = "/opt/w0034/LinuxC/SampleCode/fileDownload/send.avi";
    FILE* file = fopen(fileName, "r");
    if (NULL == file)
    {
        printf("open file failed, errno = %d.\n", errno);
        return 1;
    }

    /* 创建Socket */
    int sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (0 >= sock)
    {
        fclose(file);
        printf("socket failed, errno = %d.\n", errno);
        return 1;
    }

    /* 绑定Socket */
    struct sockaddr_in addr = {0};
    addr.sin_family = AF_INET;
    addr.sin_port = htons(8093);
    if (0 != bind(sock, (struct sockaddr*)&addr, sizeof(addr)))
    {
        fclose(file);
        printf("bind failed, errno = %d.\n", errno);
        return 1;

    }

    /* 启动监听 */
    if (0 != listen(sock, 20))
    {
        fclose(file);
        printf("listen failed, errno = %d.\n", errno);
        return 1;
    }

    /* 等待客户端连接 */
    struct sockaddr_in clientAddr = {0};
	socklen_t addrLen = sizeof(clientAddr);
    int clientSock = accept(sock, (struct sockaddr*)&clientAddr, &addrLen);

	char buf[512] = {0};
    size_t count;
    /* 读取发送文件 */
    while ((count = fread(buf, 1, 512, file)) > 0)
    {
        send(clientSock, buf, count, 0);
    }

    shutdown(clientSock, SHUT_WR);
    fclose(file);
    close(clientSock);
    close(sock);

    return 0;
}


//客户端

#include <stdio.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <errno.h>

int main(int c, char * * v)
{
    /* 输入文件名,创建文件 */
    char aucFileName[64] = {0};
    printf("请输入文件名:\n");
    scanf("%s", aucFileName);
    File* file = fopen(aucFileName, "wb");
    if (NULL == file)
    {
        printf("open file failed, errno = %d.\n", errno);
        return 1;
    }

	/* 创建Socket */
    int sock = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
    if (-1 == sock)
    {
        fclose(file);
        printf("socket failed, errno = %d.\n", errno);
        return 1;
    }

    /* 连接服务端 */
    struct sockaddr_in addr = {0};
    addr.sin_family      = AF_INET;
    addr.sin_port        = htons(8093);
    addr.sin_addr.s_addr = inet_addr("127.0.0.1");
    if (0 != connect(sock, (struct sockaddr*)&addr, sizeof(addr)))
    {
        fclose(file);
        printf("connect failed, errno = %d.\n", errno);
        return 1;
    }

    char buf[512] = {0};
    ssize_t len;
    /* 循环接收数据,写入文件 */
    while (0 != (len = recv(sock, buf, 512, 0)))
    {
        fwrite(buf, len, 1, file);
    }

    printf("file download success");
    fclose(file);
    close(sock);

    return 0;
}

Windows代码:

//服务端
#include <stdio.h>
#include <stdlib.h>
#include <winsock2.h>
#pragma comment (lib, "ws2_32.lib")  //加载 ws2_32.dll

#define BUF_SIZE 1024

int main() {
    //先检查文件是否存在
    char *filename = "D:\\send.avi";  //文件名
    FILE *fp = fopen(filename, "rb");  //以二进制方式打开文件
    if(fp == NULL){
        printf("Cannot open file, press any key to exit!\n");
        system("pause");
        exit(0);
    }

    WSADATA wsaData;
    WSAStartup( MAKEWORD(2, 2), &wsaData);
    SOCKET servSock = socket(AF_INET, SOCK_STREAM, 0);

    sockaddr_in sockAddr;
    memset(&sockAddr, 0, sizeof(sockAddr));
    sockAddr.sin_family = PF_INET;
    sockAddr.sin_addr.s_addr = inet_addr("127.0.0.1");
    sockAddr.sin_port = htons(1234);
    bind(servSock, (SOCKADDR*)&sockAddr, sizeof(SOCKADDR));
    listen(servSock, 20);

    SOCKADDR clntAddr;
    int nSize = sizeof(SOCKADDR);
    SOCKET clntSock = accept(servSock, (SOCKADDR*)&clntAddr, &nSize);

    //循环发送数据,直到文件结尾
    char buffer[BUF_SIZE] = {0};  //缓冲区
    int nCount;
    while( (nCount = fread(buffer, 1, BUF_SIZE, fp)) > 0 ){
        send(clntSock, buffer, nCount, 0);
    }

    shutdown(clntSock, SD_SEND);  //文件读取完毕,断开输出流,向客户端发送FIN包
    recv(clntSock, buffer, BUF_SIZE, 0);  //阻塞,等待客户端接收完毕

    fclose(fp);
    closesocket(clntSock);
    closesocket(servSock);
    WSACleanup();

    system("pause");
    return 0;
}


//客户端
#include <stdio.h>
#include <stdlib.h>
#include <WinSock2.h>
#pragma comment(lib, "ws2_32.lib")

#define BUF_SIZE 1024

int main(){
    //先输入文件名,看文件是否能创建成功
    char filename[100] = {0};  //文件名
    printf("Input filename to save: ");
    gets(filename);
    FILE *fp = fopen(filename, "wb");  //以二进制方式打开(创建)文件
    if(fp == NULL){
        printf("Cannot open file, press any key to exit!\n");
        system("pause");
        exit(0);
    }

    WSADATA wsaData;
    WSAStartup(MAKEWORD(2, 2), &wsaData);
    SOCKET sock = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);

    sockaddr_in sockAddr;
    memset(&sockAddr, 0, sizeof(sockAddr));
    sockAddr.sin_family = PF_INET;
    sockAddr.sin_addr.s_addr = inet_addr("127.0.0.1");
    sockAddr.sin_port = htons(1234);
    connect(sock, (SOCKADDR*)&sockAddr, sizeof(SOCKADDR));

    //循环接收数据,直到文件传输完毕
    char buffer[BUF_SIZE] = {0};  //文件缓冲区
    int nCount;
    while( (nCount = recv(sock, buffer, BUF_SIZE, 0)) > 0 ){
        fwrite(buffer, nCount, 1, fp);
    }
    puts("File transfer success!");

    //文件接收完毕后直接关闭套接字,无需调用shutdown()
    fclose(fp);
    closesocket(sock);
    WSACleanup();
    system("pause");
    return 0;
}

 

评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值