客户端:

void qt_boost::pbSendFileClicked()
{
 QString filename = ui.leFileName->text();
 QByteArray ba = filename.toLatin1();
 char * pfilename = ba.data();
 std::ifstream ifs(pfilename, ios::in|ios::binary|ios::ate);
 long size = ifs.tellg();
 char *p = new char[size];
 ifs.seekg(0, ios::beg);
 ifs.read(p, size);
 ifs.close();
 
 tcp::resolver _resolver(*ioService);
 tcp::resolver::query query(tcp::v4(), "127.0.0.1", "10800");
 tcp::resolver::iterator _iterator = _resolver.resolve(query);
 tcp::socket _socket = tcp::socket(*ioService);
 boost::asio::connect(_socket, _iterator);
 char psize[4];
 int *pint = (int *)psize;
 *pint = size;
 boost::asio::write(_socket, boost::asio::buffer(psize, sizeof(psize)));
 boost::asio::read(_socket, boost::asio::buffer(psize, sizeof(psize)));
 boost::asio::write(_socket, boost::asio::buffer(p, size));
 delete[] p;
}

调用QT的QFileDialog::getOpenFileName获取文件名称,保存在ui.leFileName控件中.使用ifstream加载文件,获取文件大小并将文件内容写入到动态数组中.向服务端发送文件大小,读取服务端回发信息后,发送文件内容.

服务端:

void session::start()
{
 //文件传输 首先接收文件大小
  socket_.async_read_some(boost::asio::buffer(data_, max_length),
  boost::bind(&session::handle_read, this, 
  boost::asio::placeholders::error,
  boost::asio::placeholders::bytes_transferred));
}

void session::handle_read(const boost::system::error_code& error, size_t bytes_transferred)
{
 if(!error)
 {
  //文件传输 接收文件大小 并回发文件大小信息 通知客户端继续发送文件内容
  int *p = (int *)data_;
        filesize = *p;
   socket_.async_write_some(boost::asio::buffer(data_, bytes_transferred),
   boost::bind(&session::handle_write, this, boost::asio::placeholders::error)); }
 else
 {
  delete this;
 }
}

void session::handle_write(const boost::system::error_code &error)
{
 if(!error)
 {

//接收文件内容
  p = new char[filesize];
  boost::asio::async_read(socket_, boost::asio::buffer(p, filesize), 
   boost::bind(&session::handle_readfile, this, boost::asio::placeholders::error));
 }
 else
 {
  delete this;
 }
}

void session::handle_readfile(const boost::system::error_code &error)
{
 if(!error)
 {

//接收完毕将文件内容存入磁盘
   std::ofstream ofile("C:\\zzz.txt");
   ofile.write(p, filesize);
   ofile.close();
 }
 else
  delete this;
}

问题:只能传输文本文件,如果要传输任何类型文件,可以修改ifstream的构造函数参数.

改进:为了可传输任意类型的文件,将文件后缀名称放在前20个字节中发送给服务端,服务端解析出后缀拼接到文件名称中.

客户端拼包代码:

QString filename = ui.leFileName->text();
 QFileInfo finfo(filename);
 QByteArray ba = filename.toLatin1();
 char * pfilename = ba.data();
 //处理后缀
 QString fileext = finfo.suffix();
 QByteArray baext = fileext.toLatin1();
 char *pfileext = baext.data();
 std::ifstream ifs(pfilename, ios::in|ios::binary|ios::ate);
 long size = ifs.tellg();
 char *p = new char[size + 20];//分配一个与文件大小相同的数组 前20个字节放文件后缀
 memset(p, 0, size + 20);
 memcpy(p, pfileext, baext.size());
 ifs.seekg(0, ios::beg);  //将当前位置置为0
 ifs.read(p + 20, size);       //将文件内容读入到数组中,待发送
 ifs.close();             //关闭文件

服务端接收代码:

void session::handle_readfile(const boost::system::error_code &error)
{
 if(!error)
 {
  char fileext[20] = {0};
  strcpy(fileext, p);
  char filename[255] = "C:\\zzz.";
  strcat(filename, fileext);
  std::ofstream ofile(filename, std::ios::out|std::ios::binary|std::ios::ate);
  ofile.write(p + 20, filesize);
  ofile.close();
 }
 else
  delete this;
}

改进中文文件名的乱码问题:

客户端读取文件流部分代码改为使用QFile类实现:

QFile file(filename);
 file.open(QIODevice::ReadOnly);
 long size = file.size();
 char *p = new char[size + 20];
 memset(p, 0, size + 20);
 memcpy(p, pfileext, baext.size());
 file.seek(0);
 file.read(p + 20, size);
 file.close();

http://blog.csdn.net/henreash/article/details/7534804

qt+boost::asio+tcp文件传输的更多相关文章

  1. 艺萌TCP文件传输及自动更新系统介绍(TCP文件传输)(四)

    艺萌TCP文件上传下载及自动更新系统介绍(TCP文件传输) 该系统基于开源的networkComms通讯框架,此通讯框架以前是收费的,目前已经免费并开源,作者是英国的,开发时间5年多,框架很稳定. 项 ...

  2. 艺萌TCP文件上传下载及自动更新系统介绍(TCP文件传输)(一)

    艺萌TCP文件上传下载及自动更新系统介绍(TCP文件传输) 该系统基于开源的networkComms通讯框架,此通讯框架以前是收费的,目前已经免费并开元,作者是英国的,开发时间5年多,框架很稳定. 项 ...

  3. boost asio tcp 多线程异步读写,服务器与客户端。

    // server.cpp #if 0 多个线程对同一个io_service 对象处理 用到第三方库:log4cplus, google::protobuf 用到C++11的特性,Windows 需要 ...

  4. boost asio tcp server 拆分

    从官方给出的示例中对于 boost::asio::ip::tcp::acceptor 类的使用,是直接使用构造函数进行构造对象,这一种方法用来学习是一个不错的方式. 但是要用它来做项目却是不能够满足我 ...

  5. Java 简单TCP文件传输

    服务端 package TCP; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputSt ...

  6. boost::asio::tcp

    同步TCP通信服务端 #include <boost/asio.hpp> #include <iostream> using namespace boost::asio; in ...

  7. Go语言 之TCP文件传输

    服务端实现流程大致如下: 创建监听listener,程序结束时关闭. 阻塞等待客户端连接,程序结束时关闭conn. 读取客户端发送文件名.保存fileName. 回发“ok”给客户端做应答 封装函数 ...

  8. boost asio tcp 多线程

    common/pools.h // common/pools.h #pragma once #include <string> #include <boost/pool/pool.h ...

  9. python基础实现tcp文件传输

    准备工作,实现文件上传需要那些工具呢? socket(传输).open()(打开文件).os(读取文件信息),当然还有辅助类sys和json,下面我们开始吧 import socket,sys imp ...

随机推荐

  1. BZOJ 3211: 花神游历各国( 线段树 )

    线段树...区间开方...明显是要处理到叶节点的 之前在CF做过道区间取模...差不多, 只有开方, 那么每个数开方次数也是有限的(0,1时就会停止), 最大的数10^9开方10+次也就不会动了.那么 ...

  2. SED修改指定行

    一个文件:cat aa #如果第三行是5的话将改为8,很明显第三行是5所以 结果改变 [root@remote ~]# sed -e '3s/5/8/' aa [root@remote ~]# #如果 ...

  3. stack around the variable “ ” was corrupted

    用scanf格式控制不当经常发生此错误. 如 short int a=10;  scanf("%d",&a); 应该是%hd; 一般是越界引起的. 参看:http://bl ...

  4. [置顶] 搭建apache+tomcat+memcached集群环境

    一.搭建apache server服务器 1.apache server图示:(加载图片要一张一张,所以可以到下面网站下载带图片的word) http://download.csdn.net/user ...

  5. 高级特性(7)- 高级AWT

    7.1 绘图操作流程7.2 形状7.3 区域7.4 笔划7.5 着色7.6 坐标变换7.7 剪切7.8 透明与组合7.9 绘图提示7.10 图像的读取器和写入器 7.10.1 获得图像文件类型的读取器 ...

  6. cocos2d-x游戏开发系列教程-中国象棋01-工程文件概述

    上一篇博文我们看到了象棋的效果图,这一张我们来看象棋代码的整体概述 让我们先对整个代码框架有个了解. 主目录: 主目录包含内容如上图: classes目录:业务代码 proj.win32:包括main ...

  7. 双缓冲绘图和窗口控件的绘制——ATL ActiveX 窗口控件生成向导绘制代码OnDraw的一个错误 .

    双缓冲绘图和窗口控件的绘制 ---ATL ActiveX 窗口控件生成向导绘制代码OnDraw的一个错误 cheungmine 我们通常使用ATL COM组件,生成一个带窗口的ActiveX控件,然后 ...

  8. 通过DWR简化AJAX开发

    DWR(Direct Web Remoting)是一个WEB远程调用框架,采取了一个类似AJAX的新方法来动态生成基于JAVA类的JavaScript代码.这样WEB开发人员就可以在JavaScrip ...

  9. boost 轻量级信号量

    #include <boost/thread/condition_variable.hpp> #include <boost/thread/mutex.hpp>     #in ...

  10. SIX GOD

    SIX GOD是什么意思呢.?_百度知道 SIX GOD