计算机网络通信程序设计-TCP文件传输程序.doc

上传人:xin****828 文档编号:6692245 上传时间:2020-03-02 格式:DOC 页数:19 大小:409.50KB
返回 下载 相关 举报
计算机网络通信程序设计-TCP文件传输程序.doc_第1页
第1页 / 共19页
计算机网络通信程序设计-TCP文件传输程序.doc_第2页
第2页 / 共19页
计算机网络通信程序设计-TCP文件传输程序.doc_第3页
第3页 / 共19页
点击查看更多>>
资源描述
目录一、实验名称:TCP文件传输程序2二、实验要求2三、总体规划21、网络传输协议的选择22、TCP协议在VC+中的实现23、传输数据的缓冲问题34、Socket的文件化管理35、数据的串行化问题46、接收数据判断是否传输完毕的方法4四、实验运行测试4五、心得体会7六、程序源代码71、建立服务器侦听套接字的类CListenSocket的定义与实现72、建立数据传输套接字的类CTransSocket的定义与实现83、用于数据串行化的类CSave的定义与实现:84、主对话框CTcpDlg类的定义与实现:9七、参考文献18一、实验名称: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连接的整个过程,编程工作量大,编程效率低,单却可以加深对网络协议的认识。程序流程示意图如下:Socket()Bind()Listen()Accept()Socket()Connect()Send()/recv()Closesocket()Send()/recv()Closesocket()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类的对象出现错误,查阅MSDN发现不能直接用 运算符不能直接对CFile类的对象进行操作。2、接收端操作同数据的传输必须同步,即必须确保在接收数据时,应确保数据已经传送到了接收端,也就是防止因为数据为传送过来而导致的接收失败。为了防止接收失败而导致数据丢失,应反复接收,直到接收数目符合为止。如:i = 0;while(i ProcessAccept();2、建立数据传输套接字的类CTransSocket的定义与实现定义部分:class CTransSocket : public Csocket/该类用于两端的连接和传输public:CTcpDlg *m_pSendDlg;public:CTransSocket(CTcpDlg *pSendDlg);virtual CTransSocket();public:virtual 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 CObjectpublic:WORD m_WEnd; /标记数据传输是否结束,结束-0 未结束-1 意外-2WORD m_WNum; /标记Bbuf中元素的个数BYTE Bbuf1024; 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+)Bbufi = 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 m_WEnd;ar m_WNum;while(i 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);elsem_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(有客户端连接);elsedelete m_pListenSocket;m_pListenSocket = NULL;delete m_pTransSocket;m_pTransSocket = NULL;JudgeButton();SetTip(CString(连接失败);void CTcpDlg:InitConnection()/初始化CSocket缓冲区及与CSocketFile /的连接。m_pFile = new CSocketFile(m_pTransSocket);m_pArchiveIn = new CArchive(m_pFile, CArchive:load,256*1024);m_pArchiveOut = new CArchive(m_pFile, CArchive:store,256*1024);void CTcpDlg:OnBUTTONListen() / TODO: Add your control notification handler code herem_pListenSocket = new CListenSocket(this);if(m_pListenSocket-Create(m_nPort)if(m_pListenSocket-Listen()JudgeButton();SetTip(CString(服务器已启动);ServerClient = 1;iEnd = 1;return;delete m_pListenSocket;m_pListenSocket = NULL;JudgeButton();SetTip(CString(服务器无法启动);ServerClient = 0;void CTcpDlg:OnBUTTONStop() iEnd = 2;delete m_pListenSocket;m_pListenSocket = NULL;delete m_pTransSocket;m_pTransSocket = NULL;JudgeButton();SetTip(CString(停止服务);ServerClient = 0;void CTcpDlg:OnBUTTONConnect() / TODO: Add your control notification handler code hereBYTE a1, a2, a3, a4;int nBlank = m_AddCtrl.GetAddress(a1,a2,a3,a4);if(nBlank Create()delete m_pTransSocket;m_pTransSocket = NULL;AfxMessageBox(创建Socket失败);return FALSE;while(!m_pTransSocket-Connect(strIP,m_nPort)if(AfxMessageBox(连接失败!您想重新尝试连接吗?,MB_YESNO) = IDNO)delete m_pTransSocket;m_pTransSocket = NULL;return FALSE;InitConnection();return TRUE;void CTcpDlg:OnBUTTONCut() / TODO: Add your control notification handler code hereiEnd = 2;delete m_pTransSocket;m_pTransSocket = NULL;JudgeButton();SetTip(CString(强制断开连接);ServerClient = 0;void CTcpDlg:ProcessRecv()/与下面的ReceiveFile()共同来接收文件CString str;double dNum = 0;DWORD num = 0;int end = 1;TRYm_pBasicFile = new CFile(m_strFileName,CFile:modeWrite|CFile:modeCreate);CATCH(CFileException, e)#ifdef _DEBUGafxDump File could not be created. m_cause Close();delete m_pBasicFile;m_pBasicFile = NULL;SendOrRecv = 1;int CTcpDlg:ReceiveFile(CFile *pBasicFile, CString strFileName, DWORD *wNum)SendOrRecv = 2;JudgeButton(); CSave Save;TRYSave.Serialize(*m_pArchiveIn);pBasicFile-Write(Save.Bbuf, sizeof(BYTE) * 1024);*wNum = Save.m_WNum;CATCH(CFileException, e)m_pArchiveOut-Abort();SendOrRecv = 0;END_CATCHif(Save.m_WEnd = 0 | Save.m_WEnd = 2)if(Save.m_WEnd = 2)if(AfxMessageBox(对方已经断开连接,文件传输中断!删除文件?,MB_YESNO) =IDYES)pBasicFile-Remove(strFileName);SendOrRecv = 0;return(2);else if(Save.m_WEnd = 0)AfxMessageBox(文件传输完毕!);SendOrRecv = 0;return(0);delete m_pArchiveIn;m_pArchiveIn = NULL;return(1);void CTcpDlg:SendFile(CString strFileName,WORD WEnd) /发送文件if(m_pArchiveOut != NULL)CString str;double dNum = 0;SendOrRecv = 1;JudgeButton();CSave Save;Save.m_WEnd = WEnd;if(!strFileName.IsEmpty()TRYm_pBasicFile = new CFile(strFileName, CFile:modeRead);CATCH( CFileException, e )#ifdef _DEBUG afxDump File could not be opened m_cause Read(Save.Bbuf,1024 * sizeof(BYTE) != 0)Save.m_WEnd = iEnd;Save.Serialize(*m_pArchiveOut);dNum = dNum + (Save.m_WNum * sizeof(BYTE) / 1024.0;str.Format(正在发送文件,已发送:%lfK字节,dNum);SetTip(str);Save.Init();m_pArchiveOut-Flush();Save.m_WEnd = 0;Save.Serialize(*m_pArchiveOut);delete m_pArchiveOut;m_pArchiveOut = NULL;AfxMessageBox(文件传输完毕!);CATCH(CFileException, e)m_pArchiveOut-Abort();delete m_pArchiveOut;m_pArchiveOut = NULL;delete m_pTransSocket;m_pTransSocket = NULL;SendOrRecv = 0;JudgeButton();CString strTemp;strTemp.Format(服务器断开连接了);SetTip(strTemp);END_CATCHdelete m_pBasicFile;m_pBasicFile = NULL;SendOrRecv = 1;elseSave.Serialize(*m_pArchiveOut);void CTcpDlg:OnBUTTONBrowser() CFileDialog * BrowserDlg;BrowserDlg = new CFileDialog(TRUE);BrowserDlg-DoModal();m_strFileName = BrowserDlg-GetPathName();m_FileName.SetWindowText(m_strFileName);delete BrowserDlg;BrowserDlg = NULL;void CTcpDlg:OnBUTTONSend() m_FileName.GetWindowText(m_strFileName);if(m_strFileName.IsEmpty()OnBUTTONBrowser();SendFile(m_strFileName,iEnd);void CTcpDlg:OnBUTTONSave() CFileDialog * SaveDlg;SaveDlg = new CFileDialog(FALSE);SaveDlg-DoModal();m_strFileName = SaveDlg-GetPathName();m_SaveFile.SetWindowText(m_strFileName); delete SaveDlg;SaveDlg = NULL;void CTcpDlg:OnBUTTONRecv() m_SaveFile.GetWindowText(m_strFileName);if(m_strFileName.IsEmpty()OnBUTTONSave(); ProcessRecv();七、参考文献Visual C+ 6.0编程实例精解 北京希望电子出版社 兰芸编著网络编程 西安理工大学计算机学院
展开阅读全文
相关资源
正为您匹配相似的精品文档
相关搜索

最新文档


当前位置:首页 > 图纸专区 > 中学资料


copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!