一篇文章讲明白Linux下的ping命令用法与实现

简介: 一篇文章讲明白Linux下的ping命令用法与实现

ping 命令用于查看网络上的主机是否在工作,它向该主机发送ICMPECHO_REQUEST 包。有时我们想从网络上的某台主机上下载文件,可是又不知道那台主机是否开着,就需要使用ping 命令查看。

ping命令的一般格式为:

ping 【-dfnqrRv】【-c 发送次数】【-i 间隔秒数】【-I 网络界面】【-l 前置载入】【-p 范本样式】【-s 数据包大小】【-t 存活数值】【主机名或IP地址】

参数说明:

-d 使用Socket的SO_DEBUG功能。

-f 极限检测。大量且快速地送网络封包给一台机器,看它的回应。

-n 只输出数值。

-q 不显示任何传送封包的信息,只显示最后的结果。

-r 忽略普通的Routing Table,直接将数据包送到远端主机上。通常是查看本机的网络接口是否有问题。

-R 记录路由过程。

-v 详细显示指令的执行过程。

-c 数目 在发送指定数目的包后停止。

-i 秒数 设定间隔几秒送一个网络封包给一台机器,预设值是一秒送一次。

-I 网络界面 使用指定的网络界面送出数据包。

-l 前置载入 设置在送出要求信息之前,先行发出的数据包。

-p 范本样式 设置填满数据包的范本样式。

-s 字节数 指定发送的数据字节数,预设值是56,加上8字节的ICMP头,一共是64ICMP数据字节。

-t 存活数值 设置存活数值TTL的大小。

:linux下的ping和windows下的ping稍有区别,linux下ping不会自动终止,需要按ctrl+c终止或者用参数-c指定要求完成的回应次数

linux下测试本机与目标主机连通性的命令是ping,这里主要讲解两个参数 –c 与 – i

其中 –c count 次数,也就是ping的次数

-i interval 间隔 ,每次ping之间的时间空格

介绍一下ping原理

ping程序实现

@头文件common.h定义,包含程序中用到的头文件及公共变量、宏、常量、静态变量定义

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#include

#define IP_HEAD_LEN 20

#define ICMP_LEN 8

#define BUFFER_SIZE 50 1024

/

原始套接字的描述符,由于需要在信号处理器

和主程序中共享,所以定义为外部变量(全局)

/

int ip_fd;

/进程号/

int p_id;

/packet_len为IP包头和ICMP包头长度之和/

extern int packet_len;

/对端地址/

struct sockaddr_in send_addr;

/发送应用缓冲区/

char send_buf【1024】;

/报文序列号/

extern int sequence;

/主机名指针/

struct hostent host;

/标识是否已经接收到回文/

int flag;

@主函数main.c定义

#include "common.h"

main(int argc, char *argv)

{

/命令是ping host(主机名)|ip_address(ip地址)/

if(argc != 2)

{

/命令不正确/

fprintf(stderr, "usage: ping .n");

exit(1);

}

/创建使用ICMP的原始套接字,这种套接字只有root才能生成/

ip_fd = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP);

if(ip_fd < 0)

{

fprintf(stderr, "raw socket error.n");

exit(1);

}

/改变进程的用户ID, 回收root权限,设置当前用户权限/

setuid(getpid());

ping(argv【1】);

}

@ping框架的建立ping.c用于初始化ping相关信息及端口信息

#include "common.h"

/

handle_alarm用于定时发送IP数据包

/

void handle_alarm(int signo)

{

send_ip();

alarm(1);

}

ping(char argv)

{

struct sigaction act;

act.sa_handler = handle_alarm;

act.sa_flags = 0;

sigemptyset(act.sa_mask);

sigaction(SIGALRM, act, NULL);

/获取main的进程id,用于设置ICMP的标志符/

p_id = getpid();

/扩大套接字接收缓冲区到50K这样做主要为了减小接收缓冲区溢出的的可能性,若无意中ping一个广播地址或多播地址,将会引来大量应答/

//setsockopt(ip_fd, SOL_SOCKET, SO_RCVBUF, BUFFER_SIZE, sizeof(BUFFER_SIZE));

/只须填写地址信息,不需要指定端口信息,因为原始套接字在传输层之下/

send_addr.sin_family = AF_INET;

/判断是主机名还是ip地址/

if(inet_addr(argv) == INADDR_NONE)

{

/是主机名/

if((host = gethostbyname(argv)) == NULL)

{

/主机名错误/

perror("get host by name error: unknow host.");

exit(1);

}

memcpy(send_addr.sin_addr, host->h_addr, host->h_length);

}

else

{

/是ip地址/

inet_aton(argv, send_addr.sin_addr);

}

printf("ping %s(%s) %d(%d) bytes of datan", argv,

inet_ntoa(send_addr.sin_addr), sizeof(struct timeval),

sizeof(struct timeval) + IP_HEAD_LEN + ICMP_LEN);

flag = 0;

/触发一个SIGALRM信号/

raise(SIGALRM);

recv_ip();

}

@发送报文send.c,建立ICMP报文并打包为IP数据包,发送

#include "common.h"

int sequence = 0;

int packet_len = IP_HEAD_LEN + ICMP_LEN;

/

send_ip用于发送包含ICMP报文的IP数据包

/

send_ip(void)

{

if(sequence != 0 !flag)

{

printf("Destination Host Unreachablen");

}

int len;

struct icmphdr icmp_p;

icmp_p = (struct icmphdr )send_buf;

/填写ICMP报文类型/

icmp_p->type = ICMP_ECHO;

/填写ICMP报文的代码/

icmp_p->code = 0;

/填写ICMP报文的标识符/

(icmp_p->un).echo.id = p_id;

/填写ICMP报文的序号,并增加ICMP的序号/

(icmp_p->un).echo.sequence = sequence ++;

/记录发送时间/

gettimeofday((struct timeval)(icmp_p + 1), NULL);

/printf("%dn", sizeof(struct icmphdr));

printf("type: %dncode: %dnchecksum: %dnun.echo.id: %dnun.echo.sequence: %dnun.gateway: %dnun.frag.unused:%d nun.frag.mtu: %dnn", icmp_p->type, icmp_p->code, icmp_p->checksum, (icmp_p->un).echo.id, (icmp_p- >un).echo.sequence, (icmp_p->un).gateway, (icmp_p->un).frag.unused, (icmp_p->un).frag.mtu);

printf("type: %dncode: %dnchecksum: %dnun.echo.id: %dnun.echo.sequence: %dnun.gateway: %dnun.frag.unused: %d nun.frag.mtu: %dnn", sizeof(icmp_p->type), sizeof(icmp_p->code), sizeof(icmp_p->checksum), sizeof((icmp_p- >un).echo.id), sizeof((icmp_p->un).echo.sequence), sizeof((icmp_p->un).gateway), sizeof((icmp_p->un).frag.unused), sizeof((icmp_p->un).frag.mtu));/

/报文的长度等于IP包长度加上ICMP报文长度和数据长度/

len = sizeof(struct timeval) + packet_len;

/使用IP计算IP包头校验和的计算方法来计算ICMP的校验和/

icmp_p->checksum = 0;

icmp_p->checksum = ip_checksum((u_short )icmp_p, len);

/发送IP数据包/

if(sendto(ip_fd, send_buf, len, 0, (struct sockaddr)send_addr, sizeof(send_addr)) < 0)

{

fprintf(stderr, "send to error.n");

}

}

@数据校验check_sum.c实现网络数据的校验工作

#include "common.h"

unsigned short ip_checksum(unsigned short pcheck, int check_len)

{

int nleft = check_len;

int sum = 0;

unsigned short p = pcheck;

//代码效果参考:http://www.zidongmutanji.com/bxxx/546600.html

unsigned short result = 0;

while(nleft > 1)

{

sum = sum + p ++;

nleft -= 2;

}

if(nleft == 1)

{

(unsigned char )(result) = (unsigned char )p;

sum += result;

}

sum = (sum ] 16) + (sum 0xFFFF);

sum += (sum ] 16);

result = sum;

return result;

}

@数据包接收receive.c,接收IP报文并分析,打印相关信息

#include "common.h"

/

recv_ip用于接受包含ICMP报文的IP数据包

/

recv_ip(void)

{

char recv_buf【1024】;

int len;

int n;

struct ip ip_p;

struct timeval time_now, time_send;

struct timeval now;

int iphead_len;

int icmp_len;

struct icmphdr icmp_p;

float delay;

while(1)

{

n = recvfrom(ip_fd, recv_buf, sizeof(recv_buf), 0, NULL, NULL);

if(n < 0)

{

if(errno = EINTR)

//代码效果参考:http://www.zidongmutanji.com/zsjx/46929.html

continue;

else

{

printf("recvfrom error.n");

continue;

}

}

ip_p = (struct ip)recv_buf;

/获取IP包头长度/

iphead_len = ip_p->ip_hl[2;

/获取IP数据包中包含的ICMP报文/

icmp_p = (struct icmphdr )(recv_buf + iphead_len);

/计算ICMP报文长度,它等于接受到的长度减去IP包头的长度/

icmp_len = n - iphead_len;

if(icmp_len < 8)

{

fprintf(stderr, "error icmp len = %d.n", icmp_len);

}

/如果ICMP类型相同则输出显示*/

if(icmp_p->type == ICMP_ECHOREPLY)

{

if((icmp_p->un).echo.id != p_id)

return;

if(icmp_len < 16)

printf("icmplen = %d.n", icmp_len);

<p style="widows: 2; text-trans

相关文章
|
20天前
|
JSON 自然语言处理 Linux
linux命令—tree
tree是一款强大的Linux命令行工具,用于以树状结构递归展示目录和文件,直观呈现层级关系。支持多种功能,如过滤、排序、权限显示及格式化输出等。安装方法因系统而异常用场景包括:基础用法(显示当前或指定目录结构)、核心参数应用(如层级控制-L、隐藏文件显示-a、完整路径输出-f)以及进阶操作(如磁盘空间分析--du、结合grep过滤内容、生成JSON格式列表-J等)。此外,还可生成网站目录结构图并导出为HTML文件。注意事项:使用Tab键补全路径避免错误;超大目录建议限制遍历层数;脚本中推荐禁用统计信息以优化性能。更多详情可查阅手册mantree。
linux命令—tree
|
23天前
|
Unix Linux
linux命令—cd
`cd` 命令是 Linux/Unix 系统中用于切换工作目录的基础命令。支持相对路径与绝对路径,常用选项如 `-L` 和 `-P` 分别处理符号链接的逻辑与物理路径。实际操作中,可通过 `cd ..` 返回上级目录、`cd ~` 回到家目录,或利用 `cd -` 在最近两个目录间快速切换。结合 Tab 补全和 `pwd` 查看当前路径,能显著提升效率。此外,需注意特殊字符路径的正确引用及脚本中绝对路径的优先使用。
|
14天前
|
Linux
Linux命令拓展:为cp和mv添加进度显示
好了,就这样,让你的Linux复制体验充满乐趣吧!记住,每一个冷冰冰的命令背后,都有方法让它变得热情起来。
40 8
|
19天前
|
安全 Linux 定位技术
Linux环境下必备的基础命令概览
以上就是Linux系统中的基本命令和工具,掌握它们就能帮你在Linux世界里游刃有余。这其实就像是学习驾驭一辆新车,熟悉了仪表盘,调整好了座椅,之后的旅程就只需要享受风驰电掣的乐趣了。
40 4
|
23天前
|
Unix Linux
linux命令—pwd
`pwd` 是 Linux/Unix 系统中的基础命令,用于显示用户当前所在的工作目录路径,帮助确认在文件系统中的位置。其核心功能包括打印逻辑路径(默认,-L 选项)和物理路径(-P 选项)。典型应用场景涵盖确认当前位置、调试符号链接问题及脚本编程中动态获取与操作路径。使用时需注意符号链接的区别、参数选择以及特殊字符处理,确保命令正确执行并满足需求。
|
2月前
|
监控 Linux
Linux命令大全:echo与tail实现输出重定向。
这样,我们实现了使用echo和tail命令进行输出重定向的目的。在实际应用中,输出重定向技巧可节省时间,提高工作效率。希望本文内容对您了解和掌握Linux系统中echo与tail命令以及输出重定向的操作有所帮助。
115 27
|
2月前
|
消息中间件 Linux Kafka
linux命令使用消费kafka的生产者、消费者
linux命令使用消费kafka的生产者、消费者
124 16
|
Linux 索引
linux命令—ls
`ls` 是 Linux 系统中用于列出目录内容的基础命令,功能强大且使用频率极高。它可以帮助用户查看文件、分析磁盘空间及检查权限等。常用选项如 `-l` 显示详细信息,`-a` 包含隐藏文件,`-h` 以易读格式展示大小,`-t` 按修改时间排序等。通过组合选项,可实现复杂需求,如递归遍历目录(`-R`)、显示 inode 号(`-i`)或结合正则过滤特定文件。注意权限限制、特殊字符处理及大规模目录操作可能带来的性能问题。掌握 `ls` 是高效使用 Linux 的关键一步。
|
1月前
|
Linux
linux文件重命名命令
本指南介绍Linux文件重命名方法,包括单文件操作的`mv`命令和批量处理的`rename`命令。`mv`可简单更改文件名并保留扩展名,如`mv old_file.txt new_name.txt`;`rename`支持正则表达式,适用于复杂批量操作,如`rename &#39;s/2023/2024/&#39; *.log`。提供实用技巧如大小写转换、数字序列处理等,并提醒覆盖风险与版本差异,建议使用`-n`参数预览效果。
|
1月前
|
安全 Linux 网络安全
Linux wget 常用命令详解
wget 是一款强大的命令行下载工具,支持 HTTP/HTTPS/FTP 协议。本文详细介绍其基础用法、高效下载参数、高级应用场景及参数速查。内容涵盖断点续传(-c)、后台下载(-b)、限速下载(--limit-rate)、递归下载(-r)、整站镜像(-mk)等实用功能,同时提供文件管理、网络优化与安全下载配置方法,助您高效完成各类下载任务。