前面介绍了TCP和UDP的通信,只是文体通信,只能传送文字。本次介绍文件传输,也就是文件读写和TCP通信的结合。

解析:根据之前的TCP通信,建立彼此的连接。服务器选择文件,首先将文件的基本信息发送给客户端。如:文件名,文件大小(用于进度条使用)。如上显示,“head#hello#1024”,即头部信息,hello就是文件名,1024是文件大小。如此客户端可以在某个磁盘创建好文件,便于后面的写入。服务端每次读取MK大小的数据,每读取一次,传输一次,客户端写入一次但要累计发送文件的大小,便于与原文件大小比较。

黏包问题:
如果头部信息和文件内容重合或追加。比如”head#hello#1024abdjsjdas”,那么“abdjsjdas”本质是内容信息而不是头部。

解决方式:设定一个定时器,在发出头部信息后,间隔N毫秒后再发送内容信息。

服务端

FileServer.h

#ifndef FILESERVER_H

#define FILESERVER_H

#include <QWidget>

#include <QTcpServer>

#include <QTcpSocket>

#include <QFile>

#include <QFileDialog>

#include <QFileInfo>

#include <QTimer>

namespace Ui {

class Widget;

}

class Widget : public QWidget

{

Q_OBJECT

public:

explicit Widget(QWidget *parent = 0);

void SendData();

~Widget();

private slots:

void on_pushButton_clicked();       //选择文件

void on_pushButton_2_clicked();     //发送文件

private:

Ui::Widget *ui;

QTcpServer *m_listenSock;   //监听套接字

QTcpSocket *m_serverSock;   //通信套接字

QFile m_file;           //文件

QString m_strFileName;  //文件名

qint64 m_fileSize;      //文件大小

qint64 m_sendSize;      //已发送的大小

QTimer timer;           //定时器

};

#endif // FILESERVER_H

FileServer.cpp

#include "FileServer.h"

#include "ui_widget.h"

#include <QMessageBox>

Widget::Widget(QWidget *parent) :

QWidget(parent),

ui(new Ui::Widget)

{

ui->setupUi(this);

this->setWindowTitle("服务器端口:888");

//将选择文件和发送文件的灰度

m_listenSock=new QTcpServer(this);

ui->pushButton->setEnabled(false);

ui->pushButton_2->setEnabled(false);

//监听

bool bIsOk=m_listenSock->listen(QHostAddress::Any,888);

if(bIsOk==true)

{

connect(m_listenSock,&QTcpServer::newConnection,

[=]()

{

//获取客户端的通信套接字

m_serverSock=m_listenSock->nextPendingConnection();

QString IP=m_serverSock->peerAddress().toString();

quint16 port=m_serverSock->peerPort();

QString strTemp=QString("[%1:%2]接入成功").arg(IP).arg(port);

ui->textEdit->setText(strTemp);

//激活选择文件按钮

ui->pushButton->setEnabled(true);

}

);

}

connect(&timer,&QTimer::timeout,

[=]()

{

//停止定时器,并开始发送文件内容

timer.stop();

SendData();

}

);

}

Widget::~Widget()

{

delete ui;

delete m_listenSock;

m_listenSock=NULL;

m_serverSock=NULL;

}

void Widget::on_pushButton_clicked()

{

QString strFilePath=QFileDialog::getOpenFileName(this,"Open","../");

if(strFilePath.isEmpty()==false)

{

//文件基本信息

QFileInfo info(strFilePath);

m_strFileName=info.fileName();

m_fileSize=info.size();

m_sendSize=0;

//qDebug()<<m_strFileName<<m_fileSize;

//指定文件名称

m_file.setFileName(strFilePath);

//打开文件

bool bIsOpen=m_file.open(QIODevice::ReadOnly);

if(bIsOpen==true)

{

ui->textEdit->append(strFilePath);

//激活发送文件按钮,灰度选择文件按钮

ui->pushButton->setEnabled(false);

ui->pushButton_2->setEnabled(true);

};

}

else

{

//打开失败

}

}

void Widget::on_pushButton_2_clicked()

{

//发送头部信息

QString strHead=QString("%1##%2")

.arg(m_strFileName)

.arg(m_fileSize);

qint64 len=m_serverSock->write(strHead.toUtf8());

//发送成功

if(len>0)

{

//防止黏包,头部信息和内容信息混淆

timer.start(1000);

}

else

{

//发送失败,重新选择文件

m_file.close();

ui->pushButton->setEnabled(true);

ui->pushButton_2->setEnabled(false);

}

}

void Widget::SendData()

{

qint64 len=0;       //每次读取的大小

do

{

//每次读取、发送4K

char pBuf[4*1024]={0};

len=0;

//读多少,发送多少

len=m_file.read(pBuf,sizeof(pBuf));

m_serverSock->write(pBuf,len);

//累加发送了多少

m_sendSize+=len;

}while(len>0);

//len>0,表明有数据读取。0数据为空,<0读取失败。

if(m_fileSize==m_sendSize)

{

//QMessageBox::information(this,"提示","发送完成!");

ui->textEdit->append("发送完成!");

//激活选择文件按钮,灰度发送文件按钮

ui->pushButton->setEnabled(true);

ui->pushButton_2->setEnabled(false);

m_file.close();

//m_serverSock->disconnectFromHost();

//m_serverSock->close();

}

}

同一程序添加两个窗口,且有设计器。新建一个文件,但不是C++类文件,而是QT且带有界面类的,最后在main中使用show()直接显示。如下图:

client.h

#ifndef CLIENT_H

#define CLIENT_H

#include <QWidget>

#include <QTcpSocket>

#include <QFile>

#include <QProgressBar>

namespace Ui {

class Client;

}

class Client : public QWidget

{

Q_OBJECT

public:

explicit Client(QWidget *parent = 0);

~Client();

private slots:

void on_pushButton_clicked();       //请求连接

void on_pushButton_2_clicked();     //关闭连接

private:

Ui::Client *ui;

QTcpSocket *m_clientSock;   //通信套接字

QProgressBar m_proBar;  //进度条

QFile m_file;           //文件

QString m_strFileName;  //文件名

qint64 m_fileSize;      //文件大小

qint64 m_recvSize;      //已接收的大小

};

#endif // CLIENT_H

client.cpp

#include "client.h"

#include "ui_client.h"

#include <QMessageBox>

#include <QFileDialog>

Client::Client(QWidget *parent) :

QWidget(parent),

ui(new Ui::Client)

{

ui->setupUi(this);

this->setWindowTitle("客户端");

m_clientSock=new QTcpSocket(this);

//初始化进度条

ui->progressBar->setValue(0);

connect(m_clientSock,&QTcpSocket::connected,

[=]()

{

ui->textEdit->setText("连接成功!");

}

);

//判断文件头部信息还是内容信息

static bool bIsHead=true;

//通信

connect(m_clientSock,&QTcpSocket::readyRead,

[=]()

{

//获取服务器发来的内容

QByteArray byteBuf=m_clientSock->readAll();

//取出文件头部信息

if(bIsHead)

{

bIsHead=false;

//"filename##9999"

//解析头部信息

m_strFileName=QString(byteBuf).section("##",0,0);

m_fileSize=QString(byteBuf).section("##",1,1).toInt();

m_recvSize=0;

//根据文件的大小,设置进度条范围

ui->progressBar->setRange(0,m_fileSize/1024);

//m_strFileName="G:\\"+m_strFileName;   //自拟文件路径

m_file.setFileName(m_strFileName);      //默认源程序上一级目录

bool bIsOpen=m_file.open(QIODevice::WriteOnly);

if(bIsOpen==false)

{

//打开失败

}

}

//将读取的内容写入新文件

else

{

//写入文件

qint64 len=m_file.write(byteBuf);

m_recvSize+=len;

//更新进度条

ui->progressBar->setValue(m_recvSize/1024);

//接收完毕

if(m_recvSize==m_fileSize)

{

QMessageBox::information(this,"提示","接收完成!");

QString strTemp=QString("%1,接收完成").arg(m_strFileName);

ui->textEdit->append(strTemp);

ui->progressBar->setValue(0);

m_file.close();

//此处一定改,不然发下一文件失败,无法获取头部信息

bIsHead=true;

}

}

}

);

}

Client::~Client()

{

delete ui;

delete m_clientSock;

m_clientSock=NULL;

}

void Client::on_pushButton_clicked()

{

//设置对方的IP和端口

QString ip=ui->IP->text();

quint16 port=ui->Port->text().toInt();

//请求连接

m_clientSock->connectToHost(ip.toUtf8(),port);

}

void Client::on_pushButton_2_clicked()

{

//断开连接

m_clientSock->disconnectFromHost();

m_clientSock->close();

}

main.cpp

#include "FileServer.h"

#include <QApplication>

#include "client.h"

int main(int argc, char *argv[])

{

QApplication a(argc, argv);

Widget w;

w.show();

Client c;

c.show();

return a.exec();

}

程序结果图:

27Tcp文件传输的更多相关文章

  1. Linux主机上实现树莓派的交叉编译及文件传输,远程登陆

    0.环境 Linux主机OS:Ubuntu14.04 64位,运行在wmware workstation 10虚拟机 树莓派版本:raspberry pi 2 B型. 树莓派OS:官网下的的raspb ...

  2. putty提供的两个文件传输工具PSCP、PSFTP详细介绍

    用 SSH 来传输文件 PuTTY 提供了两个文件传输工具 PSCP (PuTTY Secure Copy client) PSFTP (PuTTY SFTP client) PSCP 通过 SSH ...

  3. c# 局域网文件传输实例

    一个基于c#的点对点局域网文件传输小案例,运行效果截图 //界面窗体 using System;using System.Collections.Generic;using System.Compon ...

  4. 使用 zssh 进行 Zmodem 文件传输

    Zmodem 最早是设计用来在串行连接(uart.rs232.rs485)上进行数据传输的,比如,在 minicom 下,我们就可以方便的用 Zmodem (说 sz .rz 可能大家更熟悉)传输文件 ...

  5. 在windows 与Linux间实现文件传输(C++&C实现)

    要实现windows与linux间的文件传输,可以通过socket网络编程来实现. 这次要实现的功能与<Windows下通过socket进行字符串和文件传输>中实现的功能相同,即客户端首先 ...

  6. Windows下通过socket进行字符串和文件传输

    今天在windows平台下,通过socket实现了简单的文件传输.通过实现这一功能,了解基本的windows网络编程和相关函数的使用方法. 在windows平台上进行网络编程,首先都需要调用函数WSA ...

  7. Linux下几种文件传输命令 sz rz sftp scp

    Linux下几种文件传输命令 sz rz sftp scp 最近在部署系统时接触了一些文件传输命令,分别做一下简单记录: 1.sftp Secure Ftp 是一个基于SSH安全协议的文件传输管理工具 ...

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

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

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

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

随机推荐

  1. 如何在单片机上使用printf函数(printf)(avr)(stm)(lpc)(单片机)(转)

    摘要:    当我们在调试代码时,通常需要将程序中的某个变量打印至PC机上,来判断我们的程序是否按预期的运行,printf函数很好的做到了这一点,它能直接以字符的方式输出变量名和变量的值,printf ...

  2. 说说新唐ARM9(未完待续)

    针对通用32位微控制器的NUC970系列嵌入了由RISC机器有限公司设计的RISC处理器ARM926EJ-S,运行频率高达300 MHz,具有16 KB的I-cache,16 KB的D-cache和M ...

  3. dz数据结构

    pre_common_admincp_cmenu 后台 首页 | 常用操作管理数据表 字段名 数据类型 默认值 允许非空 自动递增 备注 id smallint(6) unsigned    NO 是 ...

  4. 定义了一个UIImageView如何使加载的图片不会失真 UIImageView的Frame值是固定的

    定义了一个UIImageView如何使加载的图片不会失真  UIImageView的Frame值是固定的 UIViewContentModeScaleToFill, 缩放内容到合适比例大小 UIVie ...

  5. [直观学习排序算法] 视觉直观感受若干常用排序算法 以及 iOS 资料

    http://www.zhfish.net/?s=点击范围 1 快速排序 介绍: 快速排序是由东尼·霍尔所发展的一种排序算法.在平均状况下,排序 n 个项目要Ο(n log n)次比较.在最坏状况下则 ...

  6. Mybaits中的update

    <update id="update" parameterType="Currency"> UPDATE YZ_SECURITIES_CURRENC ...

  7. java基础---->Java关于复制的使用(一)

    这里简单记录一下java中关于浅复制和深复制的知识.很多时候,一个人选择了行走,不是因为欲望,也并非诱惑,他仅仅是听到了自己内心的声音. java中的复制clone方法 一.java对象的浅复制 一个 ...

  8. Linux系统 centOS 更换软件安装源

    阿里云Linux安装软件镜像源阿里云是最近新出的一个镜像源.得益与阿里云的高速发展,这么大的需求,肯定会推出自己的镜像源.阿里云Linux安装镜像源地址:http://mirrors.aliyun.c ...

  9. TFS二次开发01——TeamProjectsPicher

    作为TFS的二次开发,首先要做的第一件工作是怎样连接到TFS并选择我们要下载的项目. 本文就此介绍一下使用TeamProjectsPicher 连接到TFS服务器. 添加引用 Microsoft.Te ...

  10. 各大互联网企业Java面试题汇总,看我如何成功拿到百度的offer

    前言 本人Java开发,5年经验,7月初来到帝都,开启面试经历,前后20天左右,主面互联网公司,一二线大公司或者是融资中的创业公司都面试过,拿了一些offer,其中包括奇虎360,最后综合决定还是去百 ...