1、C类IP地址中网络的标识长度为21位,主机标识的长度为8位,C类网络地址数量较多,适用于小规模的局域网络,每个网络最多只能包含254台计算机。C类IP地址范围192.0.1.1-223.255.254.254。D类地址(前4位是1110)用于多播(一对多通信)。E类地址(前4位是1111)保留为以后用。五类IP地址的结构如下所示:网络号(7位)主机号(24位)10网络号(14位)主机号(16位)110网络号(21位)主机号(8位)1110组播地址(28)11110保留用于课程设计和将来使用子网划分定义:Internet组织机构定义了五种IP地址,有A、B、C三类地址。A类网络有126个,每个A
2、类网络可能有16777214台主机,它们处于同一广播域。而在同一广播域中有这么多结点是不可能的,网络会因为广播通信而饱和,结果造成16777214个地址大部分没有分配出去。可以把基于类的IP网络进一步分成更小的网络,每个子网由路由器界定并分配一个新的子网网络地址,子网地址是借用基于类的网络地址的主机部分创建的。划分子网后,通过使用掩码,把子网隐藏起来,使得从外部看网络没有变化,这就是子网掩码。划分子网的方法是从网络的主机号借用若干位作为子网号subnet-id,当然主机号也就相应减少了同样的位数。IP地址=,主机号凡是从其他网络发送给本单位某个主机的IP数据报,仍然是根据IP数据报的目的网络号
3、找到连接在本单位网络上的路由器。但此路由器在收到IP数据报后,再按目的网络号和子网号找到目的子网,IP数据报交付给目的主机。三、设计方案要求捕获网络中的IP数据包,并解析,要求先进行IP数据包的捕获,先捕获再分析,这就要求用到套接字。套接字编程要求先创建原始套接字,创建原始套接字后,IP头就会包含在接收的数据中,然后对IP头进行操作,接着获取主机名和获取IP地址,并SOCKADDR_IN的结构内容进行填充,再绑定socket到本地网卡上,这就完成了IP数据包得捕获。捕获后,接收数据包,并进行分析,最后得到结果。验证ip地址合法性流程图:验证子网掩码合法性流程图得出子网号的流程主体设计流程本程序
4、主要由三部分构成:初始化原始套接字,捕获数据包和解析数据包。下面就结合核心代码对程序的具体实现进行讲解,同时使程序流程更加清晰,去掉了错误检查等保护性代码。1. 使用原始套接字套接字分为三种,即流套接字(Stream Socket)、数据报套接字(Datagram Socket)和原始套接字(Raw Socket)。要进行IP数据包的接受与发送,应使用原始套接字。创建原始套接字的代码如下:SOCKET sock;Sock=WSASoccet(AF_INET,SOCK_RAW,IPPROTO_IP,NULL,0,WSA_FLAG_OVERRLAPPED);在WSASoccet函数中,第一个参数指
5、定通信发生的区字段,AF_INET是针对Internet的,允许在远程主机之间通信。第二个参数是套接字的类型,AF_INET地址族下,有SOCK_STREAM、SOCK_DGRAM、SOCK_RAW三种套接字类型。在这里,我们设置为SOCK_RAW,表示我们声明的是一个原始套接字类型。第三个参数依赖于第二个参数,用于指定套接字所用的特定协议,这里使用IP协议。第四个参数为WSAPROTOCOL_INFO位,该位可以置空,永远置0。第六个参数是标志位,WSA_FLAG_OVERRLAPPED表明可以使用发送接收超时设置,本课程设计也可以把这个标志位设置为NULL,因为本设计不用考虑超时情况。创建
6、原始套接字后,IP头就会包含在接收的数据中。然后,我们可以设置IP头操作选项,调用sotscockpot函数。其中flag设置为TRUE,并设定IP_HDRINCL选项,表明用户可以亲自对IP头进行处理。BOOL flag=true;setsockopt (sock,IPPROTO_IP,IP_HDRINCL,(char*)&flag,sizeof(flag);之后,使用如下代码完成对socket的初始化工作:获取主机名:char hostname128;gethostname(hostname, 100);获取IP地址:hostent *pHostIP;pHostIP=gethostbyna
7、me(hostname);填充SOCKADDR_IN的结构内容:sockaddr_in addr_in;addr_in.sin_addr= *(in_addr*)pHostIP-h_addr_list0;addr_in.sin_family=AF_TNET;addr-in.sin_port=htons(6000);绑定socket: bind(sock, (POSCKADDR)&addr_in,sizeof(addr_in); 填写sockaddr_in的内容时,其地址值应填写为本机IP地址可以通过gethostbyname()函数获取;端口号可以随便填写,但不能与系统冲突;协议族应填写为AF
8、_INET。注意,sockaddr_in 结构的值必须是以网络字节顺序表示的值,而不能直接使用本机字节顺序的值,使用htoms()函数可以将无符号短整型的主机数据转换为网络字节的顺序的数据。最后使用bind()函数将socket绑定到本地网卡上。 绑定网卡后,需要WSAIoctl()函数把网卡设置为混杂模式,使网卡能够接收所有网络数据,其关键代码如下:#define SIO_RCVALL_WSAIOW(IOC_VENDOR,1)DWORD dwBufferLen10;DWORD dwBufferInLen=1;DWORD dwBytesReturned=0;WSAIoctl(SnifferSo
9、cket,IO-RCVALL,&dwBufferInLen,sizeof(dwBufferInLen),&dwBufferLen,Sizeof(dwBufferLen),&dwByteReturned,NULL,NULL); 如果接收的数据包中的协议类型和定义的原始套接字匹配,那么接收到的数据就拷贝到套接字中。因此,网卡就可以接收所有经过的IP包。2.接收数据包 在程序中可使用RECV()函数接收经过的IP包。该函数有四个参数,第一个参数接收操作所用的套接字描述符;第二个参数接收到缓冲区的地址;第二个参数接收缓冲区的地址;第三个参数接收缓冲区的大小,也就是所要接收的字节数;第四个参数是一个附加
10、标志,如果对所发送的数据没特殊要求,直接设为0。因为IP数据包的最大长度是65536B,因此缓冲区的大小不能小于65535B。设置缓冲区后,可利用循环来反复监听接收IP包,用recv()函数接收功能的代码如下:#dedine BUFFER_SIZE 65535Char bufferBUFFER_SIZE; /设置缓冲区While(true)recv(sock,buffer,BUFFER_SIZE,0); /j接收数据包 3.定义IP头部的数据结构程序需要定义一个数据结构表示IP头部。这个数据结构应该和图7-1吻合,其代码如下:typedef struct _IP_HEADER /定义IP头un
11、ionBYTE Version; /版本前4位BYTE HdrLen; /报头标长(后四位),IP头长度;BYTE ServiceType;/服务类型WORD TotalLen; /总长度WORD ID; /标识union WORD Flags; /标志Word FragOff; /分段偏移;BYTE TimeToLive; /生命期BYTE Protiocol; /协议WORD HdrChksum; /头校验和DWORD SrcAddr; /源地址DWORD DstAddr: /目的地址IP_HEADER;这是我们只考虑IP头部结构,不考虑数据部分。在捕获IP数据包后,可以通过指针把缓冲区的
12、内容强制转化为IP_HEADER数据结构。IP_HEADER ip = *( IP_HEADER *)buffer;4.IP包的解析解析IP包的字段有两种策略。针对长度为8位、16位和32位的字段 (或子字段)时,可以利用IP_HEADER的成员指教获取。要解析长度不是9位倍数的字段(或子字段)时,可以利用C语言中的位移以及与、或操作完成。下面给出了通过IP_HEADER解析IP头各个字段的代码。获取版本字段:ip.Version4;获取头部长度字段:ip.HdrLen & 0x0f;获取服务类型字段中的优先级子域:ip.ServiceType5;获取服务类型字段中的TOS子域:( IP.sE
13、RVICEtYPE1)&0X0F;获取总长度字段:ip.TotalLEN;获取标识字段:ip.ID;解析标识字段:DF=(ip.Flags14) &0x01;MF=(ip.Flags13) &0X01;获取分段偏移字段:ip.FragOff &0x1fff;获取生存时间字段:ip.TimeToLive;获取协议字段:ip.Protocol;获取头校验和字段:ip.HdrChksum;解析源IP地址字段:inet_ntoa(*(in_addr*)&ip.SrcAddr;解析目的的IP地址字段:ip.DstAddr);四、程序编写#include math.hvoid main()int m,i,
14、j,ip4,ym4,ymm48,ipp48,t,zw4,l; int ok4=0,0,0,0;printf(请输入一个IP地址:n); loop: scanf(%d.%d.%d.%d,&ip0,&ip1,&ip2,&ip3);if(ip0=0&ip0ip1ip2ip30&127) printf(此地址为合法的A类地址! else if(ip0127&192) printf(此地址为合法的B类地址! else 此地址为合法的C类地址! /判断地址的类型 /以下程序实现ip地址点分十进制到二进制的转化 for(i=0;i=0;j-) ippij=ipi%2; ipi=ipi/2;n请输入子网掩码:
15、 /将十进制子网掩码转化为二进制,存放到ym48中 scanf(ym0,&ym1,&ym2,&ym3); for(i=0; ymmij=ymi%2; ymi=ymi/2; /将相与后的数用十进制输出子网号的点分十进制表示为: for(j=0;j8;j+) ymmij=ymmij*pow(double)2,7-j); oki=ymmij+oki;%d.,oki); /将相与后的数用十进制输出五、运行结果六、实习心得和体会通过本次课程设计,通过所学的计算机网络知识,以及所查找的资料,按照要求设计出了简单的解析IP数据包的程序,从而更加深刻的了解到了IP数据包的结构及IP协议的相关问题,从而对IP层的工作原理有更好的理解和认识。对计算机网络与相结合有了一定的了解,加深对计算机网络编程的了解。在对这两项的结合中了解到了如何整合不同方面的的知识和与同学之间进行讨论交流,以获得自己所要的结果。课程设计成绩评定表学生姓名专业班级设计题目 IP数据报解析指导教师评语及意见:指导教师评阅成绩: 指导教师签字: 年 月 日注:此表装订在课程设计之后。