Linux socket编程(五) 文件传输

  这一篇给之前写的聊天室再加上文件传输:

  以下是对文件操作的封装

      FileOperator.h

#ifndef FILEOPERATOR_H
#define FILEOPERATOR_H

#include <stdio.h>

#define WRITE_CREATE_MODE "a+b"

#include <fcntl.h>
#include <string>

class FileOperator
{
    public:
        FileOperator();
        ~FileOperator();
        bool Open(const char* fileName,const char* option);
        int WriteToFile(const std::string& buffer);
        int ReadFromFile(std::string& buffer);
        void Close();

    private:
        FILE* filePtr;
};

#endif

 

FileOperator.cpp

#include "FileOperator.h"
#include <sys/sendfile.h>
#include <memory.h>

const int MAX_BUFFERLENGTH=512;

FileOperator::FileOperator()
:filePtr(NULL)
{}

FileOperator::~FileOperator()
{
    Close();
}

bool FileOperator::Open(const char* fileName,const char* option)
{
    filePtr=fopen(fileName,option);
    return filePtr!=NULL;
}


int FileOperator::WriteToFile(const std::string& buffer)
{
    int writeBytes=::fwrite(buffer.c_str(),sizeof(char),buffer.size(),filePtr);

    if(writeBytes<0)
    {
        perror("error from fwrite");
        return -1;
    }
    else if(writeBytes==0)
            return 0;
    else
        return writeBytes;
}

int FileOperator::ReadFromFile(std::string& buffer)
{
    char bufferArray[MAX_BUFFERLENGTH+1];
    buffer.clear();
    memset(bufferArray,0,MAX_BUFFERLENGTH+1);

    int numberRead=fread(bufferArray,sizeof(char),MAX_BUFFERLENGTH+1,filePtr);
    if(numberRead==-1)
    {
        perror("error in Socket::Receive");
        return -1;
    }
    else if(numberRead==0)
        return 0;
    else
    {
        buffer=bufferArray;
        return numberRead;
    }
}

void FileOperator::Close()
{
    if(filePtr!=NULL)
        fclose(filePtr);
}

那怎么发送文件呢

void ClientSocket::SendFile(const std::string& fileName)
{
    FileOperator fileOperator;
    fileOperator.Open(fileName.c_str(),"rb");
    std::string buffer;

    int readBytes;
    Send("File");

    while((readBytes = fileOperator.ReadFromFile(buffer))>0)
    {
        if(Send(buffer)<0)
        {
            perror("failed to send file");
            break;
        }
    }

}

接收文件

void ServerSocket::RecvFile(Socket* clientSocket)
{
    std::string message;
    FileOperator fileOperator;
    //using IP address to name received file
fileOperator.Open(clientSocket
->GetAddress().c_str(),WRITE_CREATE_MODE); int recvBytes; int writeBytes; while((recvBytes=Socket::Receive(*clientSocket,message))>0) { std::cout<<"message length: "<<message.size()<<"\n"; writeBytes=fileOperator.WriteToFile(message); std::cout<<"writeBytes: "<<writeBytes<<"\n"; if(writeBytes<recvBytes) { perror("write to file failed"); Socket::Send(*clientSocket,"Error when server receiving file."); return; } //if all bytes has been wrote if(recvBytes==0 || recvBytes!=MAXRECEIVE) break; } if(recvBytes >=0 ) Socket::Send(*clientSocket,"server has received your file.");
}

要注意的是发送文件时,如果文件较大,客户端很可能会阻塞,这个时候可以开一个新线程来发送文件。

不过,这个只是很简单的文件传输,复杂点的有空再慢慢研究。

转载于:https://www.cnblogs.com/-Lei/archive/2012/09/16/2687170.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
客户端,采用多线程。一个接收服务器消息,一个发送消息给服务器。 服务器,采用select()进行IO复用。 编译文件是Makefile。 (1)用户登录: 【1】client端接收用户名和密码->存于结构体中->将结构体发送给server端。 【2】server端接收client发送的结构体->打开存储用户名密码的文件->文件写入链表中->遍历链表验证用户信息。 【3】server端验证正确发送“登陆成功”消息,错误发回“登陆失败”消息。client端接收,“登陆成功”则进入聊天,“登陆失败”则跳出。 【4】若验证成功,server端产生一个新的套接字newfd,将它与用户名封装于同一个结构体中,存储在线用户的信息。 消息、存储在线用户信息结构体: typedef struct message { int type; //服务器用于判断该执行的功能 int fd; int mode; //标志位,表示用户的发言权限,1为正常,0为禁言 char name[NAMELEN]; char mima[NAMELEN]; char from[20]; char to[20]; //聊天时的收信人 char file_name[20]; //发送文件时的文件名 char mtext[100]; //聊天时发送的消息内容 struct message *next; }Mess; (2)一对多聊天: 【1】client端发送发送的信息给server端。 【2】server端遍历在线人信息链表,找到每个在线人的套接字描述符,将消息发送给链表中的每个人。 【3】可以通过输入“:)”, “:(”, “bye”来发送笑脸,悲伤脸和退出聊天;检测敏感词汇“fuck”、“shit”,禁止发送。 (3)一对一聊天: 【1】client端发送发送的信息和信息的接收者给server端。 【2】server端根据收到的接收者名字在在线人链表中查找该接收者的套接字描述符,找到后就将消息发送给该接收者。 【3】可以通过输入“:)”, “:(”, “bye”来发送笑脸,悲伤脸和退出聊天;检测敏感词汇“fuck”、“shit”,禁止发送。 (4)文件传输 【1】client端发送发送文件名和接收者名字到server端。 先打开(不存在则创建)一个文件,将文件内容读到缓冲区buffer,再将buffer的内容复制到结构体Mess中,最后将结构体发送给server端。 【2】server端先将接收到的文件重命名(因为相同文件目录下不能存在同名文件),再将收到的文件和新的文件名一同放入tab1中(并且在tab1开头写“#”)发送给client端。 【3】当client端收到以“#”开头的消息,执行文件接收,先创建一个文件,再写入相应内容。 (5)管理员模式 【1】禁言 【2】解禁
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值