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数据包 |

相关推荐
RickyWasYoung1 小时前
【代码】Matlab鸟瞰图函数
开发语言·matlab
赵英英俊7 小时前
Python day15
开发语言·python
zxsd_xyz8 小时前
基于LabVIEW与Python混合编程的变声器设计与实现
开发语言·python·labview
遇见尚硅谷8 小时前
C语言:20250712笔记
c语言·开发语言·数据结构
☞下凡☜8 小时前
C语言(20250711)
linux·c语言·开发语言
二进制person8 小时前
数据结构--准备知识
java·开发语言·数据结构
半梦半醒*8 小时前
H3CNE综合实验之机器人
java·开发语言·网络
laoliu199610 小时前
GGE Lua 详细教程
开发语言·junit·lua
勇闯逆流河10 小时前
【C++】list及其模拟实现
开发语言·c++
_Kayo_10 小时前
项目学习笔记 display从none切换成block
windows·笔记·学习