TCP文件传输程序设计书一、实验名称:TCP文件传输程序二、实验要求1、设计一个应用程序,该应用程序能够实现网络中两台计算机之间传输文件。
2、一个程序既能够建立服务器又能够以终端的形式连接服务器。
3、终端或者服务器既能够发送文件又能够接收文件。
4、传送文件类型应为任何类型,文件大小为任意。
三、总体规划1、网络传输协议的选择在TCP/IP协议栈中,有两个高级协议是我们网络应用程序编写者应该了解的,它们"传输控制协议"(Transmission Control Protocol,简称TCP)和"用户数据报协议"(User Datagrm Protocol,简称UDP)。
TCP是面向连接的通信协议,TCP提供两台计算机之间的可靠无错的数据传输。
应用程序利用TCP进行通信时,源和目标之间会建立一个虚拟连接。
这个连接一但建立,两台计算机之间就可以把数据当作一个双向字节流进行交换。
UDP是无连接通信协议,UDP不保证可靠数据的传输,但能够向若干个目标发送数据,接收发自若干个源的数据。
简单地说,如果一个主机向另外一台主机发送数据,这一数据就会立即发出,而不管另外一台主机是否已准备接收数据。
如果另外一台主机收到了数据,它不会确认收到与否。
为了使两台计算机之间传输的文件数据不会丢失或发生错误,应该采用TCP协议。
2、TCP协议在VC++中的实现在VC++中,网络协议的实现有以下几种方式:a、采用WinSocket API函数。
API函数中提供了基本Socket的系统调用,具体实现方法为服务器端首先要调用socket()函数建立一个流式套接字,用bind()函数与本机的一个端口建立关联,继续调用listen()函数将套接字置于被动的侦听方式以监听连接,然后调用accept()函数进入等待状态之后才可以接收来自客户端的请求,一旦接收到客户端通过connect发出的连接请求,accept将返回一个新的套接字描述符。
通过此套接字描述符调用send()或recv()函数即可与客户端进行数据收发。
待数据传送完成,服务器客户端调用closesocket()关闭套接字。
该方法在编程过程中需要注意socket连接的整个过程,编程工作量大,编程效率低,单却可以加深对网络协议的认识。
程序流程示意图如下:b、采用VC++中提供的MFC类,CAsyncSocket或CSocket.两个类都对WinSocket API进行了封装,CSocket对它的封装比CAsyncSocket更深,使得对于从未接触过WinSockets API的编程程序员,也能够编写网络程序。
而本程序也是采用了CSocket类进行编程。
3、传输数据的缓冲问题本机要传给对方的文件不是从外存直接通过网络发送的,而对方发送的数据也不是直接存入外存的。
而是在存中开辟一块缓冲区,从外存取出的文件先存入缓冲区,然后传给socket。
而从socket接收的数据也是先存入缓冲区然后再存到外存。
为了解决缓冲问题,VC++添加了CArchive类,CArchive类专门用来管理一块存单元,其大小可以自己来定义。
用CArhive类既以把数据载入分配的存区,又可以将存区的数据存入文件。
在该类的对象初始化时,需要和某个文件建立连接,这样数据就可以载入或存储了。
4、Socket的文件化管理在大多数编程环境和编程语言多把socket看作一个特殊的文件,其传输过程就可以看作是对文件的读写操作。
而VC++也是如此。
为了便于网络Socket的管理,在VC++中,可以对网络Socket实现文件化管理。
为了实现该功能,需要用到VC++中的类CSocketFile类,该类直接派生于CFile类,使用该类可以达到对Socket文件化管理的目的。
如CSocketFile类可以与CArchive类建立连接,这样就为Socket创立了一块缓冲区。
应该注意的是虽然CSocketFile类直接从CFile类中派生过来,但CFile类中的一些函数CSocket是不能调用的,如果调用,系统便会返回错误。
5、数据的串行化问题从对方的计算机传输过来的数据存入了存,如何将这些数据写入文件呢?要发送的文件如何将其载入存?在这个程序里我采用了数据串行化方法。
也就是通过对象的Serialize()的重载来实现文件的存取。
我在程序中采用了通过重载CObject类中的Serialize()的方式,具体做法是:从文件中读取文件数据存入数组,利用CArchive 的重载运算符 << ,将数组数据读入存,而存数据过程与其相反。
6、接收数据判断是否传输完毕的方法文件接受数据时怎样才能判断已经接受完毕呢?我采用的方法是在传输包上加标记位的方法。
每发送一个数据包,总在最前面加一个位m_WEnd,如果标记为0,说明未传输完毕,以后还有数据传送过来,如果标记为1,说明已经传输完毕,可以进行一些后续工作。
而另一端,每接受一个数据包,就检查以下该标记位,以确定是否传输完毕。
四、实验运行测试1、建立服务器2、客户端建立连接用于建立服务器用于客户端连接用于发送文件用于接收文件用于显示状态3、发送文件a、客户端发送b、服务器端接收4、成功发送5、在E:盘中查找接收到的文件此为接收到的文件五、心得体会在这五天的时间里我按照设计书的要求应用网络编程的相关知识编写了一个实现文件传输的应用程序。
在编写过程中,收获颇丰。
首先是对TCP协议和UDP协议有了更进一步的认识;其次在编写过程中,通过翻阅书籍学习了VC++编程和MFC的相关容,拓展了自己的知识面,学到了很多在课堂上无法学到的东西。
当然,由于对Socket编程毕竟还不太熟练,难免会出现一些问题,现将这些问题总结如下:1、开始时在数据串行化的设计时,直接用CArchive类的对象 << CFile类的对象或CArchive类的对象 >> CFile类的对象出现错误,查阅MSDN发现不能直接用 << 运算符不能直接对CFile类的对象进行操作。
2、接收端操作同数据的传输必须同步,即必须确保在接收数据时,应确保数据已经传送到了接收端,也就是防止因为数据为传送过来而导致的接收失败。
为了防止接收失败而导致数据丢失,应反复接收,直到接收数目符合为止。
如:i = 0;while(i < m_WNum)i = ar.Read(&Bbuf[i], m_WNum - i) + i;所幸的是这些问题都在参考资料和老师的帮助下得到了解决。
最终圆满的完成了课程设计任务书的要求。
自身分析问题和解决问题的能力也得到了提升,为以后的实验设计奠定了良好的基础。
在这里要感老师对我的悉心指导,您辛苦了!六、程序源代码1、建立服务器侦听套接字的类CListenSocket的定义与实现定义部分:class CListenSocket : public Csocket//该类用于服务器端的侦听{public:C TcpDlg *m_pSendDlg; //加该成员为了调用其的函数。
public:C ListenSocket(CTcpDlg *pSendDlg);v irtual ~CListenSocket();public:v irtual void OnAccept(int nErrorCode);};实现部分:CListenSocket::CListenSocket(CTcpDlg *pSendDlg){m_pSendDlg = pSendDlg;}void CListenSocket::OnAccept(int nErrorCode) //当服务器端收到客//户端的连接请求时执行的代码。
{C Socket::OnAccept(nErrorCode);m_pSendDlg->ProcessAccept();}2、建立数据传输套接字的类CTransSocket的定义与实现定义部分:class CTransSocket : public Csocket//该类用于两端的连接和传输{public:C TcpDlg *m_pSendDlg;public:C TransSocket(CTcpDlg *pSendDlg);v irtual ~CTransSocket();public:v irtual void OnReceive(int nErrorCode);};实现部分:CTransSocket::CTransSocket(CTcpDlg *pSendDlg){m_pSendDlg = pSendDlg;}void CTransSocket::OnReceive(int nErrorCode) //当收到发送//端发送的数据时执行的代码。
{ CSocket::OnReceive(nErrorCode);m_pSendDlg->SetTip(CString("有数据传送到"));m_pSendDlg->SendOrRecv = 2;m_pSendDlg->JudgeButton();}3、用于数据串行化的类CSave的定义与实现:定义部分:class CSave : public CObject{public:WORD m_WEnd; //标记数据传输是否结束,结束-0 未结束-1 意外-2 WORD m_WNum; //标记Bbuf[]中元素的个数BYTE Bbuf[1024];public:CSave();virtual ~CSave();void Init();virtual void Serialize(CArchive &ar);};实现部分:CSave::CSave(){Init();}void CSave::Init() //重新定义一个Init()的原因是不仅在CSave类初始{ //化时将类各变量值初始状态,还可以在以后也可int i; //以。
m_WEnd = 1; //结束标志for(i = 0; i < 1024; i++)Bbuf[i] = 0;}void CSave::Serialize(CArchive &ar) //数据串行化{unsigned int i = 0;if(ar.IsStoring()){ar << m_WEnd;ar << m_WNum;for(i = 0; i < m_WNum; i++) //数组中的值发送出ar << Bbuf[i];}else{i = 0;ar >> m_WEnd;ar >> m_WNum;while(i < m_WNum) //收到的值存入数组i = ar.Read(&Bbuf[i], m_WNum - i) + i;}}4、主对话框CTcpDlg类的定义与实现:定义部分:class CTcpDlg : public CDialog{系统定义部分省略...public:CListenSocket *m_pListenSocket;CTransSocket *m_pTransSocket;CSocketFile *m_pFile; //用于和CSocket连接CFile *m_pBasicFile;CArchive *m_pArchiveIn; //CSocket的两个缓冲区CArchive *m_pArchiveOut;CString m_strFileName; //用于存储文件名int SendOrRecv; //send 1; receive 2; nothing 0;int ServerClient; //Server-1,Client-2int iEnd; //用于控制意外中断发送信号int m_nPort;void ProcessAccept();void ProcessRecv();void InitConnection();BOOL ConnectSocket(CString strIP);void SendFile(CString strFileName, WORD WEnd = 1);int ReceiveFile(CFile *m_pBasicFile,CString strFileName, DWORD *wNum);void JudgeButton();void SetTip(CString str);public:CTcpDlg(CWnd* pParent = NULL); // standard constructorvirtual ~CTcpDlg();系统控件变量定义省略...系统定义函数省略...protected:afx_msg void OnBUTTONListen();afx_msg void OnBUTTONStop();afx_msg void OnBUTTONConnect();afx_msg void OnBUTTONCut();afx_msg void OnBUTTONBrowser();afx_msg void OnBUTTONSend();afx_msg void OnBUTTONSave();afx_msg void OnBUTTONRecv();};实现部分:系统定义函数实现省略...CTcpDlg::CTcpDlg(CWnd* pParent /*=NULL*/): CDialog(CTcpDlg::IDD, pParent){m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);m_pAutoProxy = NULL;m_pListenSocket = NULL;m_pTransSocket = NULL;m_pFile = NULL;m_pArchiveIn = NULL;m_pArchiveOut = NULL;m_strFileName.Empty();m_pBasicFile = NULL;SendOrRecv = 0;iEnd = 1;}CTcpDlg::~CTcpDlg(){if (m_pAutoProxy != NULL)m_pAutoProxy->m_pDialog = NULL;delete m_pTransSocket;m_pTransSocket = NULL;delete m_pListenSocket;m_pListenSocket = NULL;delete m_pBasicFile;m_pBasicFile = NULL;delete m_pArchiveIn;delete m_pArchiveOut;m_pArchiveIn = NULL;m_pArchiveOut = NULL;delete m_pFile;m_pFile = NULL;}BOOL CTcpDlg::OnInitDialog(){系统添加部分省略...CDialog::OnInitDialog();m_nPort = 1234;m_AddCtrl.SetAddress(192,168,0,1);JudgeButton();SetTip(CString("初始状态"));return TRUE;}void CTcpDlg::JudgeButton() //不时地调整各按钮的状态{if(m_pListenSocket == NULL && m_pTransSocket == NULL) {m_Browser.EnableWindow(TRUE);m_Listen.EnableWindow(TRUE);m_Stop.EnableWindow(FALSE);m_Connect.EnableWindow(TRUE);m_Cut.EnableWindow(FALSE);m_Send.EnableWindow(FALSE);m_Save.EnableWindow(TRUE);m_FileName.EnableWindow(TRUE);m_SaveFile.EnableWindow(TRUE);m_Recv.EnableWindow(FALSE);return;m_Recv.queque(FALSE)m_Wedive.build(FALSE)m_concent.EnableWindows(FALSE)m_agent.Forbide(TRUE)m_reduce.dosore(TRUE)}else if(m_pTransSocket != NULL){m_Listen.EnableWindow(FALSE);m_Connect.EnableWindow(FALSE);if(ServerClient == 1){m_Cut.EnableWindow(FALSE);m_Stop.EnableWindow(TRUE);}else{m_Cut.EnableWindow(TRUE);m_Stop.EnableWindow(FALSE);}if(SendOrRecv == 1){m_Recv.EnableWindow(FALSE);m_FileName.EnableWindow(TRUE);m_SaveFile.EnableWindow(FALSE);m_Browser.EnableWindow(TRUE);m_Save.EnableWindow(FALSE);}else if (SendOrRecv == 2){m_Recv.EnableWindow(TRUE);m_FileName.EnableWindow(FALSE);m_SaveFile.EnableWindow(TRUE);m_Browser.EnableWindow(FALSE);m_Save.EnableWindow(TRUE);}else{m_Recv.EnableWindow(TRUE);m_FileName.EnableWindow(TRUE);m_SaveFile.EnableWindow(TRUE);m_Browser.EnableWindow(TRUE);m_Save.EnableWindow(TRUE);}m_Send.EnableWindow(TRUE);return;}else if(m_pListenSocket != NULL){m_Browser.EnableWindow(TRUE);m_Listen.EnableWindow(FALSE);m_Stop.EnableWindow(TRUE);m_Connect.EnableWindow(FALSE);m_Cut.EnableWindow(FALSE);m_Send.EnableWindow(FALSE);m_Save.EnableWindow(TRUE);m_FileName.EnableWindow(TRUE);m_SaveFile.EnableWindow(TRUE);m_Recv.EnableWindow(FALSE);return;}}void CTcpDlg::SetTip(CString str) //调整状态区的文字显示{m_EditTip.SetWindowText(str);}void CTcpDlg::ProcessAccept() //应答客户端的连接请求{m_pTransSocket = new CTransSocket(this);if(m_pListenSocket->Accept(*m_pTransSocket)){InitConnection();m_pListenSocket->Close();delete m_pListenSocket;m_pListenSocket = NULL;JudgeButton();SetTip(CString("有客户端连接"));}else{delete m_pListenSocket;m_pListenSocket = NULL;delete m_pTransSocket;m_pTransSocket = NULL;JudgeButton();SetTip(CString("连接失败"));}}void CTcpDlg::InitConnection()//初始化CSocket缓冲区及与CSocketFile { //的连接。