两台PC之间ping通的原理.docx
《两台PC之间ping通的原理.docx》由会员分享,可在线阅读,更多相关《两台PC之间ping通的原理.docx(14页珍藏版)》请在冰豆网上搜索。
两台PC之间ping通的原理
两台PC之间ping通的原理
ICMP协议在实际传输中数据包:
20字节IP首部+8字节ICMP首部+1472字节<数据大小>38字节
ICMP报文格式:
IP首部(20字节)+8位类型+8位代码+16位校验和+(不同的类型和代码,格式也有所不同)
OSI网络七层协议
那么既然要两台PC建立ping连接必定也需要先了解OSI网络七层协议
上三层负责主机之间的应用程序通信,这三层对联网和网络地址一无所知。
接下来的四层:
传输层,网络层,数据链路层,物理层才负责联网和网络地址。
简单的解释网络七层的用途与意义
应用层:
应用层是用户与计算机交流的场所。
仅当马上需要访问网络时,这一层才会发挥作用。
简单的说:
应用层是实际应用程序之间的借口。
比如:
MicrosoftWord等应用程序并不位于应用层中,而是与应用层协议交互。
表示层:
表示层,它向应用层提供数据,并负责数据转换和代码格式化。
从本质上来说,该层是一个转换器,提供编码和转换功能。
将数据转换为标准格式在进行传输。
PC被配置成能够接受这种通用格式的数据,然后将其转换为本机格式以便读取。
表示层能够确保从一个系统的应用层传输而来的数据可被另一个系统的应用层读取。
会话层:
会话层顾名思义,就是负责在表示层实体之间建立、管理和终止会话,还对设备或节点之间的对话进行控制。
它协调和组织系统之间的通信,为此提供了三种不同的模式:
单工、半双工和全双工。
总之,会话层的基本功能是将不同应用程序的数据分离。
传输层:
传输层将数据进行分段并重组为数据流。
位于传输层的服务将来自上层应用的数据进行分段和重组,并将它们和并到同一个数据流中。
它们提供了端到端的数据传输服务,并可在互联网络上发送主机和目标主机之间建立逻辑连接。
而TCP/IP协议就在其中。
网络层:
网络层也叫IP层。
它管理设备编址、跟踪设备在网络中的位置并确定最佳的数据传输路径。
这意味着网络层必须在位于不同网络中的设备之间传输数据流。
众所周知的路由器也叫做网络层设备。
在互联网络中提供了路由选择服务。
(具体的过程在下面会提到)
数据链路层:
数据链路层提供数据的物理传输,并处理错误通知。
网络拓扑和流量控制。
这意味着数据链路层将使用硬件地址确保报文被传输到LAN中的正确设备,还将把来自网络层的报文转换为比特,供物理层传输。
数据链路层将报文封装成数据帧,并添加定制的报头,其中包含目标硬件地址和源硬件地址。
(交换机工作在数据链路层,它们根据硬件MAC地址过滤网络)
物理层:
物理层主要有两项功能:
发送和接受比特。
比特的取之智能微0或者1------使用数字值的摩尔斯码。
物理层直接与各种通信介质交流。
不同类型的介质有不同以不同的方式表示比特值。
对于每种类型的介质都需要特定的协议,这些协议描述了正确的比特模式,如何将数据编码成介质信号以及物理介质连接头的各种特征。
Ping工作过程(逻辑层次)浅
假定主机A的IP地址是192.168.1.1,主机B的IP地址是192.168.1.2,都在同一子网内,则当你在主机A上运行“Ping192.168.1.2”后,会发生了什么
首先PING是一个应用直接从应用层调用出来跳过了应用层,表示层,会话层,传输层,运行在网络层。
其次,Ping命令会构建一个固定格式的ICMP请求数据包,然后由ICMP协议将这个数据包连同地址“192.168.1.2”一起交给IP层协议(和ICMP一样,实际上是一组后台运行的进程),IP层协议将以地址“192.168.1.2”作为目的地址,本机IP地址作为源地址,加上一些其他的控制信息,构建一个IP数据包,并在一个映射表中查找出IP地址192.168.1.2所对应的物理地址(也叫MAC地址,熟悉网卡配置的朋友不会陌生,这是数据链路层协议构建数据链路层的传输单元——帧所必需的),一并交给数据链路层。
后者构建一个数据帧,目的地址是IP层传过来的物理地址,源地址则是本机的物理地址,还要附加上一些控制信息,依据以太网的介质访问规则,将它们传送出去。
其中映射表由ARP实现。
ARP(AddressResolutionProtocol)是地址解析协议,是一种将IP地址转化成物理地址的协议。
ARP具体说来就是将网络层(IP层,也就是相当于OSI的第三层)地址解析为数据连接层(MAC层,也就是相当于OSI的第二层)的MAC地址。
主机B收到这个数据帧后,先检查它的目的地址,并和本机的物理地址对比,如符合,则接收;否则丢弃。
接收后检查该数据帧,将IP数据包从帧中提取出来,交给本机的IP层协议。
同样,IP层检查后,将有用的信息提取后交给ICMP协议,后者处理后,马上构建一个ICMP应答包,发送给主机A,其过程和主机A发送ICMP请求包到主机B一模一样。
即先由IP地址,在网络层传输,然后再根据mac地址由数据链路层传送到目的主机
ICMP的应用--ping
ping可以说是ICMP的最著名的应用,当我们某一个网站上不去的时候。
通常会ping一下这个网站。
ping会回显出一些有用的信息。
一般的信息如下:
ping这个单词源自声纳定位,而这个程序的作用也确实如此,它利用ICMP协议包来侦测另一个主机是否可达。
原理是用类型码为0的ICMP发请求,受到请求的主机则用类型码为8的ICMP回应。
ping程序来计算间隔时间,并计算有多少个包被送达。
用户就可以判断网络大致的情况。
我们可以看到,ping给出来了传送的时间和TTL的数据。
我给的例子不太好,因为走的路由少,有兴趣地可以ping一下国外的网站比如,就可以观察到一些丢包的现象,而程序运行的时间也会更加的长。
ping还给我们一个看主机到目的主机的路由的机会。
这是因为,ICMP的ping请求数据报在每经过一个路由器的时候,路由器都会把自己的ip放到该数据报中。
而目的主机则会把这个ip列表复制到回应icmp数据包中发回给主机。
Ping命令的过程及返回信息分析
(自己做的图解。
)
Ping命令是我们在判断网络故障常用的命令
“Ping”的幕后过程
例如有1、2两台PC,即PC1PC2一台路由RA,子网掩码均为255.255.255.0,默认路由为192.168.0.1
1.在同一网段内
在PC1上运行“Ping192.168.1.112”后,首先,Ping命令会构建一个固定格式的ICMP请求数据包,然后由ICMP协议将这个数据包连同地址“192.168.1.112”一起交给IP层协议(和ICMP一样,实际上是一组后台运行的进程),IP层协议将以地址“192.168.1.112”作为目的地址,本机IP地址作为源地址,加上一些其他的控制信息,构建一个IP数据包,并想办法得到192.168.1.112的MAC地址(物理地址,这是数据链路层协议构建数据链路层的传输单元——帧所必需的),以便交给数据链路层构建一个数据帧。
关键就在这里,IP层协议通过PC2的IP地址和自己的子网掩码,发现它跟自己属同一网络,就直接在本网络内查找这台机器的MAC,如果以前两机有过通信,在PC1的ARP缓存表应该有PC2机IP与其MAC的映射关系,如果没有,就发一个ARP请求广播,得到B机的MAC,一并交给数据链路层。
后者构建一个数据帧,目的地址是IP层传过来的物理地址,源地址则是本机的物理地址,还要附加上一些控制信息,依据以太网的介质访问规则,将它们传送出去。
PC2收到这个数据帧后,先检查它的目的地址,并和本机的物理地址对比,如符合,则接收;否则丢弃。
接收后检查该数据帧,将IP数据包从帧中提取出来,交给本机的IP层协议。
同样,IP层检查后,将有用的信息提取后交给ICMP协议,后者处理后,马上构建一个ICMP应答包,发送给PC1,其过程和PC1发送ICMP请求包到PC2一模一样。
2.不在同一网段内
在PC1上运行“Ping192.168.1.4”后,开始跟上面一样,到了怎样得到MAC地址时,IP协议通过计算发现PC2与自己不在同一网段内,就直接将交由路由处理,也就是将路由的MAC取过来,至于怎样得到路由的MAC,跟上面一样,先在ARP缓存表找,找不到就广播吧。
路由得到这个数据帧后,再跟PC2进行联系,如果找不到,就向PC1返回一个超时的信息。
附:
一些常见的ping命令反馈的错误
1.Requesttimedout
这是对方机器置了过滤ICMP数据包,从上面工作过程来看,这是不完全正确的,至少有下几种情况。
(1)对方已关机,或者网络上根本没有这个地址:
比如在PC1中PING192.168.1.19,或者PC2关机了,在PC1中PING192.168.1.112都会得到超时的信息。
(2)对方与自己不在同一网段内,通过路由也无法找到对方,但有时对方确实是存在的,当然不存在也是返回超时的信息。
(3)对方确实存在,但设置了ICMP数据包过滤(比如防火墙设置)。
怎样知道对方是存在,还是不存在呢,可以用带参数-a的Ping命令探测对方,如果能得到对方的NETBIOS名称,则说明对方是存在的,是有防火墙设置,如果得不到,多半是对方不存在或关机,或不在同一网段内。
4)错误设置IP地址
正常情况下,一台主机应该有一个网卡,一个IP地址,或多个网卡,多个IP地址(这些地址一定要处于不同的IP子网)。
但如果一台电脑的“拨号网络适配器”(相当于一块软网卡)的TCP/IP设置中,设置了一个与网卡IP地址处于同一子网的IP地址,这样,在IP层协议看来,这台主机就有两个不同的接口处于同一网段内。
当从这台主机Ping其他的机器时,会存在这样的问题:
A.主机不知道将数据包发到哪个网络接口,因为有两个网络接口都连接在同一网段。
B.主机不知道用哪个地址作为数据包的源地址。
因此,从这台主机去Ping其他机器,IP层协议会无法处理,超时后,Ping就会给出一个“超时无应答”的错误信息提示。
但从其他主机Ping这台主机时,请求包从特定的网卡来,ICMP只须简单地将目的、源地址互换,并更改一些标志即可,ICMP应答包能顺利发出,其他主机也就能成功Ping通这台机器了。
2.DestinationhostUnreachable
(1)对方与自己不在同一网段内,而自己又未设置默认的路由,比如上例中A机中不设定默认的路由,运行Ping192.168.0.1.4就会出现“DestinationhostUnreachable”。
(2)网线出了故障
这里要说明一下“destinationhostunreachable”和“timeout”的区别,如果所经过的路由器的路由表中具有到达目标的路由,而目标因为其他原因不可到达,这时候会出现“timeout”,如果路由表中连到达目标的路由都没有,那就会出现“destinationhostunreachable”。
3.BadIPaddress
这个信息表示您可能没有连接到DNS服务器,所以无法解析这个IP地址,也可能是IP地址不存在。
4.Sourcequenchreceived
这个信息比较特殊,它出现的机率很少。
它表示对方或中途的服务器繁忙无法回应。
5.Unknownhost——不知名主机
这种出错信息的意思是,该远程主机的名字不能被域名服务器(DNS)转换成IP地址。
故障原因可能是域名服务器有故障,或者其名字不正确,或者网络管理员的系统与远程主机之间的通信线路有故障。
6.Noanswer——无响应
这种故障说明本地系统有一条通向中心主机的路由,但却接收不到它发给该中心主机的任何信息。
故障原因可能是下列之一:
中心主机没有工作;本地或中心主机网络配置不正确;本地或中心的路由器没有工作;通信线路有故障;中心主机存在路由选择问题。
7.Ping127.0.0.1:
127.0.0.1是本地循环地址
如果本地址无法Ping通,则表明本地机TCP/IP协议不能正常工作。
8.norouttohost:
网卡工作不正常。
9.transmitfailed,errorcode:
10043网卡驱动不正常。
10.unknownhostname:
DNS配置不正确。
那么到这里仅仅是ping的逻辑上的原理。
由浅及深,那么物理上到底从什么开始运行的呢
网上搜索ping的物理上的原理的文章少之又少可以说是几乎没有。
唯独有一篇写到了。
ping运行使用到了ping.c的一个内核编译文件(C语言写的)
而在linux下面ping.c是使用C语言编译写下并赋予在内核中的文件中,封装起来了所以在linux系统中并不能找到ping.c文件
那么由ping.c开始搜索。
并找到了ping.c的源码
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#definePACKET_SIZE4096
#defineMAX_WAIT_TIME5
#defineMAX_NO_PACKETS3
charsendpacket[PACKET_SIZE];
charrecvpacket[PACKET_SIZE];
intsockfd,datalen=56;
intnsend=0,nreceived=0;
structsockaddr_indest_addr;
第四步接收无限循环消息其中最重要的就是调用到了cyclie函数
其中包含了cycliewait客户端回应消息,形成ICMP回应包,设定等待时间,调用select函数。
发送signt调用carcher,调用SIGALARM信号来结束程序。
其中的Pinger函数也是最重要的一个,它是为了形成ICMP回应数据包(填写icmphdr结构体的数据),pr_pack数据包显示函数它可以将数据的显示部分独立出来,加强程序的逻辑性和结构性。
最后的Finish程序可以打印出一些统计信息。
pid_tpid;
structsockaddr_infrom;
structtimevaltvrecv;
voidstatistics(intsigno);
unsignedshortcal_chksum(unsignedshort*addr,intlen);
intpack(intpack_no);
voidsend_packet(void);
voidrecv_packet(void);
intunpack(char*buf,intlen);
voidtv_sub(structtimeval*out,structtimeval*in);
voidstatistics(intsigno)
{printf("\n--------------------PINGstatistics-------------------\n");
printf("%dpacketstransmitted,%dreceived,%%%dlost\n",nsend,nreceived,
(nsend-nreceived)/nsend*100);
close(sockfd);
exit
(1);
}
参考文献:
逻辑