收藏 分销(赏)

基于UDP协议的网络聊天程序设计.doc

上传人:丰**** 文档编号:4324020 上传时间:2024-09-06 格式:DOC 页数:16 大小:215.14KB 下载积分:8 金币
下载 相关 举报
基于UDP协议的网络聊天程序设计.doc_第1页
第1页 / 共16页
基于UDP协议的网络聊天程序设计.doc_第2页
第2页 / 共16页


点击查看更多>>
资源描述
《基于UDP协议的网络聊天程序设计》 基于UDP协议的网络聊天程序设计 学生姓名:XX 指导老师:XX 摘 要 本课程设计是是基于UDP协议的网络聊天程序设计,UDP协议是无连接的协议,是TCP/IP协议的一种分层协议。通过本设计进一步掌握利用Visual C++进行程序设计的能力和运用面向对象程序设计的思想和方法,初步掌握开发一个小型实用系统的基本方法,理解Windows程序的运行过程。本设计实现的聊天工具要求在Microsoft Visual C++ 6.0上,编写一个程序,采用C/S模式,模拟网络上的聊天软件,实现在局域网上两台主机之间的通信。且经过测试,该聊天程序能够正常运行,实现了设计目标。 关键词 UDP协议;客户机/服务器;网络聊天 1 引 言 现在是网络的世界,网络发展的不可估量了,它可以包括我们生活中的方方面面,小到上网浏览新闻、上网购物,大到网上理财、网上会议,可以说网络把我们的世界变小了,即使在海角天边,只要有网络的地方我们就可以随时联系。最简单的形式就是进行网络聊天了,比如QQ、MSN等等很简单的聊天工具,可以随时拉近你我的距离。本设计将基于UDP协议和局域网环境,实现一个即时聊天程序。 1.1本文主要内容 本文第二节介绍了UDP协议和客户机/服务器模式基本原理,第三节详细描述了设计思路和流程,第四节为本次课程设计的总结。 1.2设计平台 Microsoft Visual C++ 6.0,简称VC或者VC6.0,是微软推出的一款C++编译器,将“高级语言”翻译为“机器语言(低级语言)”的程序。Visual C++是一个功能强大的可视化软件开发工具。自1993年Microsoft公司推出Visual C++1.0后,随着其新版本的不断问世,Visual C++已成为专业程序员进行软件开发的首选工具。虽然微软公司推出了 Visual C++.NET(Visual C++7.0),但它的应用有很大的局限性,只适用于Windows 2000、Windows XP和Windows NT4.0。所以实际中,更多的是以Visual C++6.0为平台。 Visual C++6.0由Microsoft开发, 它不仅是一个C++ 编译器,而且是一个基于Windows操作系统的可视化集成开发环境(integrated development environment,IDE)。Visual C++6.0由许多组件组成,包括编辑器、调试器以及程序向导AppWizard、类向导Class Wizard等开发工具。 这些组件通过一个名为Developer Studio的组件集成为和谐的开发环境。Microsoft的主力软件产品。Visual C++是一个功能强大的可视化软件开发工具。自1993年Microsoft公司推出Visual C++1.0后,随着其新版本的不断问世,Visual C++已成为专业程序员进行软件开发的首选工具。虽然微软公司推出了Visual C++.NET(Visual C++7.0),但它的应用的很大的局限性,只适用于Windows 2000,Windows XP和Windows NT4.0。所以实际中,更多的是以Visual C++6.0为平台。 2 设计原理 2.1 UDP协议原理 UDP协议的全称是用户数据报协议,在网络中它与TCP协议一样用于处理数据包,是一种无连接的协议。在OSI模型中,在第四层——传输层,处于IP协议的上一层。UDP有不提供数据包分组、组装和不能对数据包进行排序的缺点,也就是说,当报文发送之后,是无法得知其是否安全完整到达的。UDP用来支持那些需要在计算机之间传输数据的网络应用。包括网络视频会议系统在内的众多的客户/服务器模式的网络应用都需要使用UDP协议。UDP协议从问世至今已经被使用了很多年,虽然其最初的光彩已经被一些类似协议所掩盖,但是即使是在今天UDP仍然不失为一项非常实用和可行的网络传输层协议。 UDP是OSI参考模型中一种无连接的传输层协议,它主要用于不要求分组顺序到达的传输中,分组传输顺序的检查与排序由应用层完成,提供面向事务的简单不可靠信息传送服务。UDP 协议基本上是IP协议与上层协议的接口。UDP协议适用端口分别运行在同一台设备上的多个应用程序。 与所熟知的TCP(传输控制协议)协议一样,UDP协议直接位于IP(网际协议)协议的顶层。根据OSI(开放系统互连)参考模型,UDP和TCP都属于传输层协议。 UDP协议的主要作用是将网络数据流量压缩成数据包的形式。一个典型的数据包就是一个二进制数据的传输单位。每一个数据包的前8个字节用来包含报头信息,剩余字节则用来包含具体的传输数据。 UDP是无连接的,即发送数据之前不需要建立连接(当然发送数据结束时也没有连接可释放)因此减少了开销和发送数据之前的时延。 UDP使用的是尽最大努力交付,即不保证可靠交付,因此主机不需要维持复杂的连接状态表。 UDP是面向报文的。发送方的UDP对应程序交下来的报文,在添加首部后就向下交付给IP层。UDP对应用层交下来的报文,既不合并,也不拆分,而是保留这些报文的边界。 UDP是定义用来在互连网络环境中提供数据报交换的计算机通信的协议。此协议默认是IP下层协议。此协议提供了向另一用户程序发送信息的最简便的协议机制,不需要连接确认和保护复制,所以在软件实现上比较简单,需要的内存空间比起TCP相对较小。 UDP包头由4个域组成,其中每个域各占用2个字节。 (1)源端口号(16位):UDP数据包的发送方使用的端口号。 (2)目标端口号(16位):UDP数据包的接收方使用的端口号。UDP协议使用端口号为不同的应用保留其各自的数据传输通道。UDP和rap协议正是采用这一机制,实现对同一时刻内多项应用同时发送和接收数据的支持。 (3)数据报长度(16位)。数据报的长度是指包括报头和数据部分在内的总的字节数。理论上,包含报头在内的数据包的最大长度为65535字节。不过,一些实际应用往往会限制数据包的大小,有时会降低到8192字节。 (4)校验值(16位)。UDP协议使用包头中的校验值来保证数据的安全。 2.2 客户机/服务器模式   由于这次课程设计是实现点到点的可靠连接,所以在此使用UDP连接方式。在这个连接中,双方分为客户和服务器,他们各自的功能不同。 客户机一方,UDP应用程序功能如下: (1) 打开通信信道(申请一套接字),并连接到服务器在主机的保留端口,该端口对应服务器的UDP进程。 (2) 向服务器发出请求报文,等待接收应答。 (3) 从服务器方收到最终应答结果,或在不再请求时关闭信道并终止客户机进程。 服务器一方,UDP应用程序动作如下: (1) 打开通信信道(申请一套接字),通知本地主机在某一保留端口接收客户机请求。 (2) 等待客户请求到达指定端口。 (3) 接收到请求,启动一新进程处理用户请求,同时释放旧进程以响应新的客户请求,一旦服务完成,关闭新进程与客户的通信链路。 (4) 继续等待客户机请求。 (5) 如果不想响应客户机请求,关闭服务器进程。 3设计步骤 3.1 总体设计思路 基于UDP协议通信和基于TCP协议的通信不同,基于UDP的消息传递更快,但不提供可靠性的保证。也就是说,数据在传递的时候,用户无法知道数据能否正确的到达目的地主机,也不能确定数据到达目的地的顺寻是否和发送的顺序相同。系统采用的是客户端/服务器工作模式,所以应有客户端和服务器两个类,其中客户端向服务器请求服务。而基于UDP通信的基本模式是: 将数据打包,称为数据包,然后将数据包发往目的地。 接受别人发来的数据包,然后查看数据包的内容。 3.2功能模块调用流程设计: 客户端 服务器 Socket() Socket() Bind() Bind() 双向 通信 ReceiveFrom() /SendTo() ReceiveFrom() /SendTo() Close() Close() 图3.1 程序设计流程图 其中,服务器端:Socket(),建立流式套接字;Bind(),将套接字和服务器地址结构绑定;ReceiveFrom()/SendTo(),在套接字上接收/发送数据;Close(),关闭套接字。客户端与服务器端同理。 3.3基于UDP的网络聊天程序实现 3.3.1 基于UDP的聊天程序服务器端源程序: //创建套接字 //第二个参数和TCP设置不同 SOCKET sockSrv = socket(AF_INET, SOCK_DGRAM, 0); SOCKADDR_IN addrSrv; //设置服务器端套接字的地址结构的相关属性 addrSrv.sin_addr.S_un.S_addr = htonl(INADDR_ANY); //设置IP addrSrv.sin_port = htons(20000); // //设置端口号 addrSrv.sin_family = AF_INET; //将套接字和服务器地址结构绑定 bind(sockSrv, (SOCKADDR *)&addrSrv, sizeof(SOCKADDR) ); char recvBuf[100]; char sendBuf[100]; char tempBuf[100]; SOCKADDR_IN addrClient; int len = sizeof(SOCKADDR); //从客户端接收数据 recvfrom(sockSrv, recvBuf, 100, 0, (SOCKADDR *)& addrClient, & len); if ('q' == recvBuf[0]) { sendto(sockSrv, "q", strlen("q") + 1, 0, (SOCKADDR*) & addrClient, len); cout << "chart end" << endl; break; //退出循环 } //将对方发送过来的信息前段加上IP地址进行输出 sprintf(tempBuf, "%s say %s\n", inet_ntoa(addrClient.sin_addr), recvBuf); cout << tempBuf << endl; //将用户的键盘输入发送到对方 cout << "Please input message: " << endl; gets(sendBuf); sendto(sockSrv, sendBuf, strlen(sendBuf) + 1, 0, (SOCKADDR * )&addrClient, len); 3.3.2 基于UDP的聊天程序客户端端源程序: //创建套接字 SOCKET sockClient = socket(AF_INET, SOCK_DGRAM, 0); //第二个参数和TCP设置不同 SOCKADDR_IN addrSrv; //服务器地址结构 addrSrv.sin_addr.S_un.S_addr = inet_addr("127.0.0.1"); //服务器地址 addrSrv.sin_port = htons(20000); //服务器端口号 addrSrv.sin_family = AF_INET; cout << "请输入发送的消息: " << endl; gets(sendBuf); //得到用户的键盘输入 //将对方发送过来的信息前段加上IP地址进行输出 sprintf(tempBuf, "%s say %s\n", inet_ntoa(addrSrv.sin_addr), recvBuf); cout << tempBuf << endl; 3.4程序测试结果 图3.2 客户端向服务器发送信息 图3.3 服务器收到信息 图3.4 服务器向客户端发送信息 图3.5 客户端收到服务器信息 4结束语 这次计算机网络课程设计历时两个星期,让我学到了很多很多的东西。不仅巩固了以前所学过的知识,而且学到了很多在书本上所没有学到过的知识。当然,在这个过程中我也遇到了很多问题,不过由于老师耐心的帮助和讲解,使我更进一步加深了对计算机网络的了解,让我对它有了更加浓厚的兴趣。特别是当每一步成功时,心里特别的开心。 总的来说,这次设计的基于UDP协议的网络聊天程序还是比较成功的,在设计中遇到了很多问题,最后在老师的辛勤指导下,终于迎刃而解,有点小小的成就感,终于觉得平时所学的知识有了实用的价值,达到了理论与实际相结合的目的,不仅学到了不少知识,而且锻炼了自己的能力,使自己对以后的路有了更加清楚的认识,同时,对未来有了更多的信心。最后,对给过我帮助的所有同学和各位指导老师再次表示忠心的感谢! 参考文献 [1] 谢希仁.计算机网络(第四版)[M].北京:电子工业出版社,2003 [2] 陆魁军 等.计算机网络基础实践教程[M].北京:清华大学出版社,2005 [3]陈坚、陈伟编著.《Visual C++网络高级编程》. 北京: 人民邮电出版社,2001 [4]吴功宜、胡晓英等编著.《计算机网络课程设计》.北京:机械工业出版社,2010.8 附件 1 : 首先编写服务器端程序,实现代码如下: Chat Server #include <iostream> #include <WinSock2.h> #pragma comment(lib, "Ws2_32.lib") using namespace std; int main(int argc, CHAR * argv[]) { WORD wVersionRequested; WSADATA wsaData; int err; wVersionRequested = MAKEWORD(2, 0); err = WSAStartup(wVersionRequested, & wsaData); if (err != 0 ) { cout << "WSAStartup Error!" << endl; return 1; } if (LOBYTE(wsaData.wVersion) != 2 || HIBYTE(wsaData.wVersion) != 0) { cout << "Version not rignt!" << endl; WSACleanup(); return 1; } //创建套接字 //第二个参数和TCP设置不同 SOCKET sockSrv = socket(AF_INET, SOCK_DGRAM, 0); SOCKADDR_IN addrSrv; // //设置服务器端套接字的地址结构的相关属性 addrSrv.sin_addr.S_un.S_addr = htonl(INADDR_ANY); //设置IP addrSrv.sin_port = htons(20000); // //设置端口号 addrSrv.sin_family = AF_INET; //将套接字和服务器地址结构绑定 bind(sockSrv, (SOCKADDR *)&addrSrv, sizeof(SOCKADDR) ); char recvBuf[100]; char sendBuf[100]; char tempBuf[100]; SOCKADDR_IN addrClient; int len = sizeof(SOCKADDR); while(1) { //从客户端接收数据 recvfrom(sockSrv, recvBuf, 100, 0, (SOCKADDR *)& addrClient, & len); if ('q' == recvBuf[0]) { sendto(sockSrv, "q", strlen("q") + 1, 0, (SOCKADDR*) & addrClient, len); cout << "chart end" << endl; break; //退出循环 } //将对方发送过来的信息前段加上IP地址进行输出 sprintf(tempBuf, "%s say %s\n", inet_ntoa(addrClient.sin_addr), recvBuf); cout << tempBuf << endl; //将用户的键盘输入发送到对方 cout << "Please input message: " << endl; gets(sendBuf); sendto(sockSrv, sendBuf, strlen(sendBuf) + 1, 0, (SOCKADDR * )&addrClient, len); } closesocket(sockSrv); WSACleanup(); return 0; } 下面编写聊天客户端程序,实现代码如下: Chat client #include <iostream> #include <WinSock2.h> #pragma comment(lib, "Ws2_32.lib") //静态加入一个lib文件 using namespace std; int main(int argc, CHAR * argv[]) { WORD wVersionRequested; WSADATA wsaData; int err; wVersionRequested = MAKEWORD(2, 0); err = WSAStartup(wVersionRequested, & wsaData); if (err != 0 ) { cout << "WSAStartup Error!" << endl; return 1; } if (LOBYTE(wsaData.wVersion) != 2 || HIBYTE(wsaData.wVersion) != 0) { cout << "Version not rignt!" << endl; WSACleanup(); return 1; } //创建套接字 SOCKET sockClient = socket(AF_INET, SOCK_DGRAM, 0); //注意第二个参数和TCP设置不同 SOCKADDR_IN addrSrv; //服务器地址结构 addrSrv.sin_addr.S_un.S_addr = inet_addr("127.0.0.1"); //服务器地址 addrSrv.sin_port = htons(20000); //服务器端口号 addrSrv.sin_family = AF_INET; char recvBuf[100]; char sendBuf[100]; char tempBuf[100]; int len = sizeof(SOCKADDR); while(1) { cout << "请输入发送的消息: " << endl; gets(sendBuf); //得到用户的键盘输入 sendto(sockClient, sendBuf, strlen(sendBuf) + 1, 0, (SOCKADDR * )&addrSrv, len); recvfrom(sockClient, recvBuf, 100, 0, (SOCKADDR *)& addrSrv, & len); if ('q' == recvBuf[0]) { sendto(sockClient, "q", strlen("q") + 1, 0, (SOCKADDR *) & addrSrv, len); cout << "chart end" << endl; break; //退出循环 } //将对方发送过来的信息前段加上IP地址进行输出 sprintf(tempBuf, "%s say %s\n", inet_ntoa(addrSrv.sin_addr), recvBuf); cout << tempBuf << endl; } closesocket(sockClient); WSACleanup(); return 0; } 第16页 共16页
展开阅读全文

开通  VIP会员、SVIP会员  优惠大
下载10份以上建议开通VIP会员
下载20份以上建议开通SVIP会员


开通VIP      成为共赢上传

当前位置:首页 > 学术论文 > 其他

移动网页_全站_页脚广告1

关于我们      便捷服务       自信AI       AI导航        抽奖活动

©2010-2026 宁波自信网络信息技术有限公司  版权所有

客服电话:0574-28810668  投诉电话:18658249818

gongan.png浙公网安备33021202000488号   

icp.png浙ICP备2021020529号-1  |  浙B2-20240490  

关注我们 :微信公众号    抖音    微博    LOFTER 

客服