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

相关推荐
喜欢喝果茶.2 分钟前
QOverload<参数列表>::of(&函数名)信号槽
开发语言·qt
亓才孓3 分钟前
[Class类的应用]反射的理解
开发语言·python
努力学编程呀(๑•ี_เ•ี๑)3 分钟前
【在 IntelliJ IDEA 中切换项目 JDK 版本】
java·开发语言·intellij-idea
island131424 分钟前
CANN GE(图引擎)深度解析:计算图优化管线、内存静态规划与异构任务的 Stream 调度机制
开发语言·人工智能·深度学习·神经网络
坚持就完事了28 分钟前
Java中的集合
java·开发语言
魔芋红茶32 分钟前
Python 项目版本控制
开发语言·python
wjhx37 分钟前
QT中对蓝牙权限的申请,整理一下
java·数据库·qt
踏过山河,踏过海38 分钟前
【qt-查看对应的依赖的一种方法】
qt·visual studio
云小逸1 小时前
【nmap源码解析】Nmap OS识别核心模块深度解析:osscan2.cc源码剖析(1)
开发语言·网络·学习·nmap
冰暮流星1 小时前
javascript之二重循环练习
开发语言·javascript·数据库