QT 在Windows下实现ping功能(ICMP)

前言

很多时候,我们可能会图省事直接调用系统中的ping命令,但这是很不科学的~

废话不多说,直接上代码..

.pro文件

在.pro文件末尾添加一行:

复制代码
LIBS += -liphlpapi -lws2_32

.h文件

在.h文件中加入:

复制代码
#include <QDebug>
#include <winsock2.h>
#include <iphlpapi.h>
#include <icmpapi.h>
#include <ws2tcpip.h>
#include <iostream>
using namespace std;

private:
    int ping(const char *ip); #也可以加在public下,不是一定要private

.cpp文件:

cpp 复制代码
// ping函数
int Widget::ping(const char *ip){
    //通常使用MAKEWORD宏来指定版本,如MAKEWORD(2, 2)表示使用Winsock 2.2版本。
    WSADATA wsaData;

    //WSAStartup函数用于初始化Winsock库,并指定Winsock库的版本。这个函数需要在使用Winsock库的其他函数之前调用,而在程序结束时,你需要调用WSACleanup函数来释放Winsock库的资源。
    if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
    {
        qDebug()<<"Error1:"<<WSAGetLastError();
        return -1;
    }

    // 创建socket
    SOCKET sock = socket(AF_INET, SOCK_RAW, IPPROTO_ICMP);
    if (sock == INVALID_SOCKET)
    {
        qDebug()<<"Error2:"<<WSAGetLastError();
        return -2;
    }

    //设置目的地址
    SOCKADDR_IN dest;
    dest.sin_family = AF_INET;
    dest.sin_addr.s_addr = inet_addr(ip);

    //创建ICMP数据包
    const int packet_size = 32; //数据包大小,取值范围 0-65507
    char packet[packet_size];
    memset(packet, 0, packet_size); //memset函数通常用于初始化一段内存空间,或者将一段内存空间清空。这句的作用是将packet指向的内存空间清空,以便后续的代码可以使用这段内存空间来存储ICMP报文

    //创建icmp句柄
    ICMP_ECHO_REPLY reply;
    HANDLE icmp_handle = IcmpCreateFile();
    if (icmp_handle == INVALID_HANDLE_VALUE)
    {
        qDebug()<<"Error3:"<<GetLastError();
        return -3;
    }

    DWORD reply_size = packet_size + 28; //接收到的ICMP回复数据的大小。+28等同于:+sizeof(reply)

    //发送ICMP数据包
    int status = IcmpSendEcho(icmp_handle, dest.sin_addr.s_addr, packet, packet_size, NULL, &reply, reply_size, 2000);

    if (status == 0) //这里status == 0 表示执行失败,要查看错误码可以查看: reply.Status
    {
        qDebug()<<"Error4:"<<GetLastError();
        return -4;
    }

    // Print the response
    qDebug()<<"延时:" << reply.RoundTripTime; //查看延时
    qDebug()<<"包大小:" << reply.DataSize; //查看数据包大小
    qDebug()<<"TTL值:" << (int)reply.Options.Ttl; //查看ttl值

    WSACleanup(); //释放WSAStartup

    return reply.RoundTripTime; //返回延时
}

ping函数调用方法示例:

方法一:

cpp 复制代码
ping("192.168.1.1");

方法二:

cpp 复制代码
QString ip = "192.168.1.1";
ping(ip.toUtf8());

如果要重复执行,可以通过循环重复调用ping函数..

reply.Status状态码:

|--------------------------|-----------|---------------------------|
| 常量名 | | 含义 |
| IP_SUCCESS | 0 | 状态是成功。 |
| IP_BUF_TOO_SMALL | 11001 | 答复缓冲区太小。 |
| IP_DEST_NET_UNREACHABLE | 11002 | 目标网络不可达。 |
| IP_DEST_HOST_UNREACHABLE | 11003 | 目标主机不可达。 |
| IP_DEST_PROT_UNREACHABLE | 11004 | 目的地的协议是遥不可及。 |
| IP_DEST_PORT_UNREACHABLE | 11005 | 目标端口不可达。 |
| IP_NO_RESOURCES | 11006 | IP资源不足。 |
| IP_BAD_OPTION | 11007 | 指定了错误的IP选项。 |
| IP_HW_ERROR | 11008 | 一个硬件错误。 |
| IP_PACKET_TOO_BIG | 11009 | 包太大。 |
| IP_REQ_TIMED_OUT | 11010 | 请求超时。 |
| IP_BAD_REQ | 11011 | 一个坏的请求。 |
| IP_BAD_ROUTE | 11012 | 一个糟糕的路线。 |
| IP_TTL_EXPIRED_TRANSIT | 11013 | 在传输过程中的生存时间(TTL)的过期。 |
| IP_TTL_EXPIRED_REASSEM | 11014 | 在碎片重组过程中的生存时间过期。 |
| IP_PARAM_PROBLEM | 11015 | 一个参数的问题。 |
| IP_SOURCE_QUENCH | 11016 | 数据报到达太快,处理和数据报可能被丢弃。 |
| IP_OPTION_TOO_BIG | 11017 | 一个IP选项是太大了。 |
| IP_BAD_DESTINATION | 11018 | 一个坏的目的地。 |
| IP_GENERAL_FAILURE | 11050 | 一般故障。这个错误可以返回一些畸形的ICMP数据包 |

相关推荐
三行数学几秒前
Matlab之父克利夫·莫勒尔逝世
开发语言·matlab
陌路202 分钟前
C++高级进阶--夯实进阶基础(1)
开发语言·c++
梦想三三19 分钟前
【PYthon词频统计与文本向量化】苏宁易购评论分析实战
开发语言·python
AI人工智能+电脑小能手21 分钟前
【大白话说Java面试题 第93题】【Mysql篇】第23题:从查找速度来看,聚集索引和非聚集索引哪个更快?
java·开发语言·数据库·mysql·面试
Cheng小攸1 小时前
入侵检测环境部署
开发语言·php
Smoothcloud润云1 小时前
5大功能精修,重构AI算力使用体验!
java·人工智能·windows·算法·重构·编辑器·sublime text
我是唐青枫1 小时前
Java MyBatis-Flex 实战指南:从 BaseMapper 到 QueryWrapper 的轻量 ORM 用法
java·开发语言·mybatis
ModestCoder_2 小时前
windows/ubuntu解决挂梯子但是codex reconnecting五次的问题
linux·windows·ubuntu
ShyanZh2 小时前
Markitdown 多格式文档智能解析实战指南
开发语言·c#
一只专注api接口开发的技术猿2 小时前
OpenClaw 对接淘宝商品 API,低成本实现全天候选品监控|附可运行 Python 实操代码
大数据·开发语言·数据库·python