




已阅读5页,还剩17页未读, 继续免费阅读
版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
滁州学院课程设计报告课程名称: 计算机网络课程设计 设计题目: ping程序的设计与实现 系 别: 计算机与信息工程学院 专 业: 计算科学与技术 组 别: 第五小组 起止日期: 2011年12月1日2011年12月8日 指导教师: 计算机科学与技术系二一一年制课程设计题目Ping程序的设计与实现组长学号2011220125班级计专(2)班系别计算机与信息工程学院专业计算机科学与技术组员指导教师课程设计目的通过设计Ping程序,理解Ping程序的实现原理,并初步讲解了c语言网络编程技术。本章涉及很多网络编程函数和编程技巧,包括库文件的导入;winsock的初始化、注销;socket的创建、关闭;设置socket选项;根据主机名获取IP地址; 从堆中分配一定数量的空间、释放从堆中分配的空间;数据报的发送;数据报的接等。课程设计所需环境WindowsXP+Visual C+6.0课程设计任务要求实现ping的基本功能,实现ping -t课程设计工作进度计划序号起止日期工 作 内 容分工情况012011-12-12011-12-2展开思路讨论工作并搜集相关资料022011-12-32011-12-6具体制作,编写相关代码,制作相关窗口并实现,美化界面。032011-12-72011-12-8编写并完成课程设计报告指导教师签字: 年 月 日教研室审核意见:教研室主任签字: 年 月 日课程设计任务书一Ping 程序运行原理在网络层, 除了IP协议之外, 还有一些控制协议, 如ICMP, ARP, DHCP等。1. ping的基础知识原始套接字原始套接字是允许访问底层传输协议的一种套接字类型。使用原始套接字操作IP数据报, 可以进行路由跟踪, Ping等。另外, 使用原始套接字需要知道许多下层协议结构的知识,所以下面讨论ICMP,IP, UDP, TCP格式。原始套接字有两种类型, 第一种类型是在IP头种使用预定义的协议, 如ICMP;第二种类型是在IP头种使用自定义的协议。下面使用创建原始套接字的方法。创建套接字的函数是socket()或者WSASocket(),只不过要将套接字类型指定为SOCK_RAW,代码如下:SOCKET sraw = :socket(AF_INET, SOCK_RAW, IPPROTO_ICMP);创建原始套接字时socket函数的第三个参数protocol值将成为IP头中得协议域的值。IPPROTO_ICMP指定要使用ICMP。原始套解释提供管理下层传输的能力。他们可能会被恶意利用, 因此, 仅Administrator组的成员能够创建SOCK_RAW类型的套接字。 任何人在Windows NT下都可以创建原始套接字,但是没有Administrator权限的人不能用它来做任何事情, 因为bind函数将会失败, 出错码WSAEACCES.在上面的套接字创建代码种,我们使用ICMP,也可以使用IIGMP, UDP, IP或者原始IP,对应的宏定义分别是IPPROTO_IGMP, IPROTO_UDP, IPPROTO_IP或者IPPROTO_RAW。其中协议标志IPPROTO_UDP, IPPROTO_IP, 和IPPROTO_RAW 需要启动IP_HDRINCL选项。使用恰当的协议标志创建原始套接字之后,便可以在发送和接受调用种使用此套接字句柄了。无论IP_HDRINCL选项是否设置, 在原始套接字上接收到的数据种都会将包含IP头。2. ICMP协议与校验和的计算互联网上得操作由路由器紧紧地监控着。当有异常饭送时候,具体事件通过ICMP报道,如目的不可到达,TTL超时等。这个协议也用来测试互联网。每个ICMP消息都封装在IP封包中, 所以使用IP寻址,ICMP消息的格式如下:首8位表示ICMP的类型,通常可以分为请求消息和错误报告消息两类。接下来的八位表示ICMP代码,这个域进一步定义了请求或者是消息的类型。接下来八位表示icmp的校验和。它提供了ICMP头和他的实际数据。3.校验和的计算发送ICMP报文时, 必须由程序自己计算校验和,并将它填入ICMP头部的对应域中。校验和的计算方法是:将数据以资为单位累加到一个双字中,如果数据长度为奇数,最后一个字节将被扩展到字, 累加的结果是一个双字,最后将这个双字的高16bit和低16bit相加后取反,便得到了校验和。u_short checksum(u_short *buffer, int len)register int nleft = len;register u_short *w = buffer;register u_short answer;register int sum = 0;/使用32位累加器,进行16位的反馈计算while ( nleft 1 )sum += *w+;nleft -= 2;/补全奇数位if ( nleft = 1 )u_short u = 0;*(u_char *)(&u) = *(u_char*)w;sum += u;/将反馈的16位从高位移到低位sum = (sum 16) + (sum & 0xffff);sum += (sum 16);answer = sum;return (answer);3.Ping程序设计思路:要实现ping程序,需要实现以下步骤:(1) 创建协议类型为IPPROTO_ICMP的原始套接字,设置套接字属性。(2) 创建并初始化ICMP封包。(3) 调用sendto函数向远程主机发送ICMP请求。(4) 调用recfrom函数接受ICMP响应。初始化ICMP头时先初始化消息类型和代码域, 之后是回显请求头。程序首先定义了ICMP头的数据结构IMCP_HDR.。ICMP_HDR的定义如下:typedef struct _ICMPHeaderu_char Type; /类型u_char Code; /代码u_short Checksum; /首部校验和u_short ID; /标识u_short Seq; /序列号char Data; /数据ICMPHDR, *PICMPHDR;4.编程时,需要用到一些windows函数,说明如下:(1).int WSAStartup(WORD wVersionRequested,LPWSADATA lpWSAData);函数说明:为了在应用程序当中调用任何一个Winsock API函数,首先第一件事情就是必须通过WSAStartup函数完成对Winsock服务的初始化,因此需要调用WSAStartup函数。使用Socket的程序在使用Socket之前必须调用WSAStartup函数。该函数的第一个参数指明程序请求使用的Socket版本,其中高位字节指明副版本、低位字节指明主版本;操作系统利用第二个参数返回请求的Socket的版本信息。当一个应用程序调用WSAStartup函数时,操作系统根据请求的Socket版本来搜索相应的Socket库,然后绑定找到的Socket库到该应用程序中。以后应用程序就可以调用所请求的Socket库中的其它Socket函数了。(2).SOCKET socket( int af, int type, int protocol ); 函数说明:应用程序调用socket函数来创建一个能够进行网络通信的套接字。 第一个参数指定应用程序使用的通信协议的协议族,对于TCP/IP协议族,该参数置AF_INET; 第二个参数指定要创建的套接字类型,流套接字类型为SOCK_STREAM、数据报套接字类型为SOCK_DGRAM、 原始套接字SOCK_RAW(WinSock接口并不适用某种特定的协议去封装它,而是由程序自行处理数据包以 及协议首部); 第三个参数指定应用程序所使用的通信协议。(3).int sendto( SOCKET s, const char FAR *buf, int len, int flags, const struct sockaddr FAR *to, int tolen);函数说明:返回值:实际发送数据的长度。 parameter : s 套接字 buff 待发送数据的缓冲区 size 缓冲区长度 Flags 调用方式标志位, 一般为0, 改变Flags,将会改变Sendto发送的形式 addr (可选)指针,指向目的套接字的地址 len addr所指地址的长度(4)int recvfrom(SOCKET s, char FAR* buf, int len, int flags, struct sockaddr FAR *from, int FAR *fromlen );函数说明:recvfrom( )用来接收远程主机经指定的socket传来的数据,并把数据传到由参数buf指向的内存空间,参数len为可接收数据的最大长度.参数flags一般设0,其他数值定义参考recv().参数from用来指定欲传送的网络地址,结构sockaddr请参考bind()函数.参数fromlen为sockaddr的结构长度.二程序的流程图和源码Ping程序的设计与实现大致可分为四个模块(见图1-1),分别是:初始化模块、功能控制模块、ping模块、mian测试模块。Ping程序的设计与实现初始化模块功能控制模块Ping模块main测试模块图1-11.初始化模块:该模块用于定义及初始化各个全局变量,为winsock加载winsock体。(见图1-2)主要包括定义IP首部格式、定义ICMP首部格式、定义ICMP回应请求、定义ICMP回应答复。初始化模块定义IP首部格式定义ICMP首部格式定义ICMP回应请求定义ICMP回应答复图1-22.功能控制模块:该模块是被其他模块调用,其功能包括计算校验和、发送回应请求函数、接收应答回复并进行解析、等待回应答复(主是要select模型)。(见图1-3,1-4,1-5,1-6)计算校验和函数源码:否 (将反馈的16位从高位移到低位)是(使用32位累加器,进行16位的反馈计算)Checksum()开始定义初始化一些变量Nleft1sum += *w+;nleft -= 2;Nleft=1是(补全奇数位)sum += u;sum = (sum 16) + (sum & 0xffff);sum += (sum 16);answer = sum;返回answer结束图1-3是SendEchoRequest()开始定义初始化一些变量填充要发送的数据存储发送的时间计算回应请求的校验和发送回应请求返回nRet调用发送错误函数结束图1-4WaitForEchoReply ()开始定义初始化一些变量返回timeout结束是RecvEchoReply ()开始定义初始化一些变量接收应答回复检验接收结果返回应答时间调用发送错误函数结束图1-6图1-53.Ping模块功能模块:该模块是本程序的核心模块,调用其他模块实现其功能,进而实现Ping的功能。s否是是开始定义初始化各个局部变量判断WSAGetLastError()是否调用成功否检测目标主机是否为NULL设置目标主机的IP地址,开始ping发起四次ping测试发送ICMP回应请求等待回复的数据接收回复计算花费时间Loop是否为0 输出平均次数输出ping结果清除残余结束4.main()函数模块:向指定的域名或IP地址发送Echo 请求报文;根据响应报文显示出Ping的结果;程序仅支持-t选项即可。否开始WSAStartup() 是否成功截取后三位字符调用ping()释放资源加载失败结束为了实现-t图1-8三运行操作及结果在VC中运行程序后会出现如图4.1所示,提示你输入IP或网址;2、我们先输入校园网机房主机命令,看能否ping通3.试着使用ping t命令,如下:4、再输入外部网主机命令,看能否ping通,上图为网络ping不通的情形。源代码源代码如下:#include #include #include #include#pragma comment(lib, ws2_32.lib)/导入库文件#define ICMP_ECHOREPLY 0 /ICMP回应答复#define ICMP_ECHOREQ 8 /ICMP回应请求#define REQ_DATASIZE 32 /请求数据报大小#include /*/using namespace std;/定义IP首部格式typedef struct _IPHeader u_char VIHL; /版本和首部长度u_char ToS; /服务类型u_short TotalLen; /总长度u_short ID; /标识号u_short Frag_Flags; /片偏移量u_char TTL; /生存时间u_char Protocol; /协议u_short Checksum; /首部校验和struct in_addr SrcIP; /源IP地址struct in_addr DestIP; /目的地址IPHDR, *PIPHDR;/定义ICMP首部格式typedef struct _ICMPHeaderu_char Type; /类型u_char Code; /代码u_short Checksum; /首部校验和u_short ID; /标识u_short Seq; /序列号char Data; /数据ICMPHDR, *PICMPHDR; /定义ICMP回应请求typedef struct _ECHOREQUESTICMPHDR icmpHdr;DWORD dwTime;char cDataREQ_DATASIZE;ECHOREQUEST, *PECHOREQUEST;/定义ICMP回应答复typedef struct _ECHOREPLYIPHDR ipHdr;ECHOREQUEST echoRequest;char cFiller256;ECHOREPLY, *PECHOREPLY;/*/计算校验和u_short checksum(u_short *buffer, int len)register int nleft = len;register u_short *w = buffer;register u_short answer;register int sum = 0;/使用32位累加器,进行16位的反馈计算while ( nleft 1 )sum += *w+;nleft -= 2;/补全奇数位if ( nleft = 1 )u_short u = 0;*(u_char *)(&u) = *(u_char*)w;sum += u;/将反馈的16位从高位移到低位sum = (sum 16) + (sum & 0xffff);sum += (sum 16);answer = sum;return (answer);/*/发送回应请求函数int SendEchoRequest(SOCKET s, struct sockaddr_in *lpstToAddr)static ECHOREQUEST echoReq;static nId = 1;static nSeq = 1;int nRet;/填充回应请求消息echoReq.icmpHdr.Type = ICMP_ECHOREQ;echoReq.icmpHdr.Code = 0;echoReq.icmpHdr.Checksum = 0;echoReq.icmpHdr.ID = nId+;echoReq.icmpHdr.Seq = nSeq+;/填充要发送的数据for (nRet = 0; nRet h_addr); /设置目标IPdestIP.sin_family = AF_INET; /地址规格destIP.sin_port = 0;/提示开始进行PINGprintf(nPinging %s %s with %d bytes of data:n,pstrHost,inet_ntoa(destIP.sin_addr),REQ_DATASIZE);/发起多次PING测试for (nLoop=0; nLoopmaximum)maximum=dwElapsed;if(dwElapsedminimum)minimum=dwElapsed;average+=dwElapsed;printf(nReply from %s: bytes = %d time = %ldms TTL = %d, inet_ntoa(srcIP.sin_addr),REQ_DATASIZE,dwElapsed,cTTL);if(_kbhit() /* Use _getch to throw key away. */ if (c=_getch()=0x2) /crrl -b break; else Sleep(1000);printf(nn);printf(Ping statistics for %s:n,inet_ntoa(srcIP.sin_addr);printf( Packets: Sent = %d, Received = %d, Lost = %d (%.f% loss),n,sent,reveived,lost,(float)(lost*1.0/sent)*100);if(lost=0)printf(Approximate round trip times in milli-seconds:n); printf( Minimum = %dms, Maximum = %dms, Average = %dmsn,minimum,maximum,average/sent);printf(nn);nRet = closesocket(rawSocket);if (nRet = SOCKET_ERROR)printf(closesocket() error:%dn, WSAGetLastError();/主程序void main()printf(Welcome to the Ping Testn);while(1)WSADATA wsd;/检测输入的参数/初始化Winsockif (WSAStartup(MAKEWORD(1, 1), &wsd) != 0)/第一个函数说明 WSAStartup()printf(加载 Winsock 失败!n);char opt1100;char *ptr=opt1;bool log=false;printf(Ping );cin.getline(opt1,100,n);/ping的地址 字符串if(strstr(opt1, -t)!=NULL)log=true;strncpy(ptr,opt1+0,strlen(opt1)-3);/把原字符串的最后三位截取ptrstrlen(opt1)-2=0; /printf(%s, ptr);/开始PINGPing(ptr,log);/程序释放 Winsock 资源WSACleanup();/*函数说明为了在应用程序当中调用任何一个Winsock API函数,首先第一件事情就是必须通过WSAStartup函数完成对Winsock服务的初始化,因此需要调用WSAStartup函数。使用Socket的程序在使用Socket之前必须调用WSAStartup函数。该函数的第一个参数指明程序请求使用的Socket版本,其中高位字节指明副版本、低位字节指明主版本;操作系统利用第二个参数返回请求的Socket的版本信息。当一个应用程序调用WSAStartup函数时,操作系统根据请求的Socket版本来搜索相应的Socket库,然后绑定找到的Socket库到该应用程序中。以后应用程序就可以调用所请求的Socket库中的其它Socket函数了。*/*函数说明SOCKET socket( int af, int type, int protocol ); 应用程序调用socket函数来创建一个能够进行网络通信的套接字。 第一个参数指定应用程序使用的通信协议的协议族,对于TCP/IP协议族,该参数置AF_INET; 第二个参数指定要创建的套接字类型,流套接字类型为SOCK_STREAM、数据报套接字类型为SOCK_DGRAM、 原始套接字SOCK_RAW(WinSock接口并不适用某种特定的协议去封装它,而是由程序自行处理数据包以 及协议首部); 第三个参数指定应用程序所使用的通信协议。心得体会通过仔细阅读程序代码,查找相关资料,我大概弄懂了程序的基本过程。程序通过发送一个ICMP回显请求报文到目的地主机,如果有IP选项途中的主机通过报文记录各自的IP地址,目的地主机回发一个回显应答报文,然后发送主机通过解析回显应答报文,查看通过路由地址,并计算发送回收报文所用的传送时间,以确定回显报文是否超时。程序中很多算法值得我们借鉴,例如检验和算法先把两数通过取反相加,然后再次取反以防止上次取反数组溢出而出现错误。我们从中要学会代码的重复使用,程序中usage(char *progname)反复使用来输出信息,而不用每次输出相关的信息的时候都需要重新写实现代码,这样既节省程序编译运行所需的时间和内存的开销又易于以后程序的升级和添加其他功能,符合编程
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 社会治理与公共政策互动研究试题及答案
- 项目管理中的人际关系与沟通试题及答案
- 计算机三级软件测试技能要求解析试题及答案
- 2025年初级银行从业资格(银行管理)通关练习题库
- 计算机测试过程中的数据管理试题及答案
- 机电工程职能定位试题及答案
- 项目团队合作的成功案例分析试题及答案
- 行业动态对考试知识点的影响试题及答案
- 机电工程2025年基本技能试题及答案
- 2025年成人继续教育线上学习模式下的学习效果提升与教学平台功能优化研究报告
- GB/T 3452.4-2020液压气动用O形橡胶密封圈第4部分:抗挤压环(挡环)
- 西南科技大学机械原理期末考试复习题及答案
- 读后续写:骑马迷路 遇困难不放弃 课件 【知识建构+点播拓展】高考英语作文备考
- 2023年宜兴市云湖茶禅文旅发展有限公司招聘笔试题库及答案解析
- 初中地理会考知识点汇总
- Unit2Reading2知识点课件-高中英语牛津译林版(2020)选择性必修第一册
- 交通协管员劳务外包服务方案
- 顶管工程顶进记录表
- 安全生产、环境保护监督管理制度(最终版)
- 呼吸道病原体抗体检测及临床应用课件
- 战略管理教学ppt课件(完整版)
评论
0/150
提交评论