检查 Windows 计算机是否通过 C++ 连接到以太网

Check if Windows Computer on Ethernet Via C++

因此,我尝试查看了几个不同的答案。我认为可能有希望的是这个:

How to check network interface type is Ethernet or Wireless on Windows using Qt?

但是,我真的不太了解网络,甚至 Windows。就个人而言,我无法理解他们网站上的大部分 Microsoft 文档。我已经尝试过 INetworkConnectionNativeWiFi 等方法。但要么他们没有按照我的意愿行事,要么我只是无法从可用文档中弄清楚如何做。

话虽如此,我想使用 C++ 检查运行此程序的设备 运行 是否通过以太网电缆连接到互联网。基本上,我想执行以下操作:

但是,问题是我不知道如何检查设备是否连接了以太网。有没有办法做到这一点?我没有使用 QT。谢谢!


编辑:我还应该包括到目前为止我尝试过的内容。

我尝试使用 GetAdaptersInfo 并从 PIP_ADAPTER_INFO 变量类型中获取 Type 特征,但无论我是否在以太网上,这总是给我 Unknown type 71

GetAdaptersInfo 的文档在这里: https://msdn.microsoft.com/en-us/library/aa365917%28VS.85%29.aspx

谢谢


编辑 2:这是我用于 GetAdaptersInfo 的代码

bool is_on_ethernet{
    PIP_ADAPTER_INFO pAdapterInfo;
    PIP_ADAPTER_INFO pAdapter = NULL;
    DWORD dwRetVal = 0;
    UINT i;

    struct tm newtime;
    char buffer[32];
    errno_t error;

    ULONG ulOutBufLen = sizeof(IP_ADAPTER_INFO);
    pAdapterInfo = (IP_ADAPTER_INFO *)malloc(sizeof(IP_ADAPTER_INFO));

    if(pAdapterInfo == NULL)
        printf("Error allocating memory need to call GetAdaptersInfo");

    if(GetAdaptersInfo(pAdapterInfo, &ulOutBufLen) == ERROR_BUFFER_OVERFLOW){
        free(pAdapterInfo);
        pAdapterInfo = (IP_ADAPTER_INFO *)malloc(ulOutBufLen);
    }

    if((dwRetVal = GetAdaptersInfo(pAdapterInfo, &ulOutBufLen)) == NO_ERROR){
        pAdapter = pAdapterInfo;

        switch(pAdapter->Type){
            case MIB_IF_TYPE_OTHER:
                printf("Other\n");
                return false;
                break;
            case MIB_IF_TYPE_ETHERNET:
                printf("Ethernet\h");
                return true;
                break;
            case MIB_IF_TYPE_TOKENRING:
                printf("Token Ring\n");
                return false;
                break;
            case MIB_IF_TYPE_FDDI
                printf("FDDI\n");
                return false;
                break;
            case MIB_IF_TYPE_PPP
                printf("PPP\n");
                return false;
                break;
            case MIB_IF_TYPE_LOOPBACK
                printf("Lookback\n");
                return false;
                break;
            case MIB_IF_TYPE_SLIP
                printf("Slip\n");
                return false;
                break;
            default
                printf("Unknown type %ld\n\n", pAdapter->Type);
                return false;
                break;
        }
    }

    if(pAdapterInfo)
        free(pAdapterInfo);

    return false;
}

首先,非常感谢用户@Nighthawk441 为我指明了正确的方向。没有这个用户,我肯定想不出解决方案。

话虽这么说,但我现在拥有的解决方案充其量只是一个 hack。它似乎有效,但我认为它甚至不是最佳选择。因此,我将把它留作答案,但如果找到更好的答案,我暂时不会接受它。我也非常愿意接受任何人对这个答案的任何评论。

简而言之,我所做的是遍历 GetAdaptersInfo 中的所有适配器。为了查看适配器是否已连接,我将适配器的 IP 地址与字符串 "0.0.0.0" 进行了比较,好像它不是这个我觉得说适配器已连接是安全的。所以,事不宜迟,这是我实现的代码。

bool is_on_ethernet(){
    PIP_ADAPTER_INFO pAdapterInfo;
    PIP_ADAPTER_INFO pAdapter = NULL;
    DWORD dwRetVal = 0;
    UINT i;

    struct tm newtime;
    char buffer[32];
    errno_t error;

    ULONG ulOutBufLen = sizeof(IP_ADAPTER_INFO);
    pAdapterInfo = (IP_ADAPTER_INFO *)malloc(sizeof(IP_ADAPTER_INFO));

    if(pAdapterInfo == NULL)
        printf("Error allocating memory needed to call GetAdaptersInfo");

    if(GetAdaptersInfo(pAdapterInfo, &ulOutBufLen) == ERROR_BUFFER_OVERFLOW){
        free(pAdapterInfo);
        pAdapterInfo = (IP_ADAPTER_INFO *)malloc(ulOutBufLen);
    }

    if((dwRetValue = GetAdaptersInfo(pAdapterInfo, &ulOutBufLen)) == NO_ERROR){
        do{
            pAdapter = pAdapterInfo;

            string ip_string = pAdapter->IpAddressList.IpAddress.String;

            switch(pAdapter->Type){
                case MIB_IF_TYPE_OTHER:
                    printf("Other\n");
                    break;
                ...
                case MIB_IF_TYPE_ETHERNET:
                    printf("Ethernet\n");

                    //new code
                    if(ip_string.compare("0.0.0.0") != 0){
                        free(pAdapterInfo);
                        return true;
                    }

                    break;
                default:
                    printf("Unknown type %ld\n", pAdapter->Type);
                    break;
            }
        }while(pAdapterInfo = pAdapterInfo->Next);
    }

    if(pAdapterInfo)
        free(pAdapterInfo);

    return false;
}

查看此参考资料对我很有帮助:

https://msdn.microsoft.com/en-us/library/windows/desktop/aa365819%28v=vs.85%29.aspx

非常感谢 Nighthawk 向我提供这些信息。希望这会帮助别人!如果有人有任何意见或任何其他答案,请随时 post 他们!谢谢!

您的问题有点困难,因为获取 "current" 网络适配器可能真的很复杂 --- windows 根据网络适配器配置和目标可达性路由数据包,因此您的 "current" 适配器可能随时更改...但由于您已经知道如何检索可用适配器的 IP 和 MACs ("hardware address"),您可以简单地使用 hack 来检索 MAC对于你当前的 IP 和 filter/search 里面我的第二个功能吧!字段 "PhysicalAddress" 就是您要查找的内容,即 MAC 地址

我的经验是,唯一可靠的方法是通过 GetIfTable and GetIfTable2, the former returns somewhat superficial adpater info and the latter provides great detail。 这是一个示例实现,因为它使用了详细的功能,您还可以查询 WLAN 适配器:

vector<MIB_IF_ROW2>* getDevices(NDIS_PHYSICAL_MEDIUM type)
    {       
        vector<MIB_IF_ROW2> *v = new vector<MIB_IF_ROW2>();
        PMIB_IF_TABLE2 table = NULL;
        if(GetIfTable2Ex(MibIfTableRaw, &table) == NOERROR && table)
        {
            UINT32 i = 0;
            for(; i < table->NumEntries; i++)
            {
                MIB_IF_ROW2 row;

                ZeroMemory(&row, sizeof(MIB_IF_ROW2));
                row.InterfaceIndex = i;
                if(GetIfEntry2(&row) == NOERROR)
                {                   
                    if(row.PhysicalMediumType == type)
                    {
                        v->push_back(row);
                    }                   
                }           
            }
            FreeMibTable(table);
        }
        return v;
    }

现在您需要做的就是遍历列表并过滤掉禁用的适配器和诸如此类的东西:

vector<MIB_IF_ROW2>* wlan = getDevices(NdisPhysicalMediumNative802_11); //WLAN adapters
//see https://msdn.microsoft.com/en-us/library/windows/desktop/aa814491(v=vs.85).aspx, "PhysicalMediumType" for a full list
for(auto &row : *v)
    {
        //do some additional filtering, this needs to be changed for non-WLAN           
        if( row.TunnelType == TUNNEL_TYPE_NONE &&
            row.AccessType != NET_IF_ACCESS_LOOPBACK &&         
            row.Type == IF_TYPE_IEEE80211 &&
            row.InterfaceAndOperStatusFlags.HardwareInterface == TRUE)              
            {
                //HERE BE DRAGONS!                    
            }
    }

现在很容易生成 WLAN 适配器和非 WLAN 适配器列表 (请参阅第二个函数中的评论),搜索您当前的 MAC 并得出结论它是有线或无线的 - 但请注意这些列表 可能 重叠,因为 802.11 basically is an extended version of 802.3 802.3 包括 802.11 (因为它是一个扩展) - 所以你需要一点点 if/else 逻辑来将 WLAN 与非 WLAN 适配器分开。

您也可以使用 WlanEnumInterfaces 获取所有 WLAN 适配器,但这与使用上述函数并以 NdisPhysicalMediumNative802_11 作为参数基本相同 ...

根据@specializt 的回答,经过一些小的修改,我得到了如下适用的方法:

BOOL getDevices(NDIS_PHYSICAL_MEDIUM type, vector<MIB_IF_ROW2>& vetIFRow)
{
    PMIB_IF_TABLE2 table = NULL;
    if (GetIfTable2Ex(MibIfTableRaw, &table) != NOERROR || !table)
    {
        return FALSE;
    }

    for (ULONG i = 0; i < table->NumEntries; i++)
    {
        MIB_IF_ROW2 row;
        ZeroMemory(&row, sizeof(MIB_IF_ROW2));
        row.InterfaceIndex = i;
        if (GetIfEntry2(&row) == NOERROR && row.PhysicalMediumType == type)
        {
            vetIFRow.push_back(row);
        }
    }

    FreeMibTable(table);
    return TRUE;
}

BOOL isNetIFConnected(const MIB_IF_ROW2& row, IFTYPE Type)
{
    return (row.TunnelType == TUNNEL_TYPE_NONE &&
        row.AccessType != NET_IF_ACCESS_LOOPBACK &&
        row.Type == Type &&
        row.InterfaceAndOperStatusFlags.HardwareInterface == TRUE &&
        row.MediaConnectState == MediaConnectStateConnected);
}

tstring getNetWorkType()
{
    vector<MIB_IF_ROW2> vectRow;
    BOOL bRet = getDevices(NdisPhysicalMedium802_3, vectRow); // ETHERNET adapters
    if (bRet)
    {
        for (auto it = vectRow.begin(); it != vectRow.end(); it++)
        {
            if (isNetIFConnected(*it, IF_TYPE_ETHERNET_CSMACD))
            {
                return L"ETHERNET";
            }
        }
    }

    vectRow.clear();
    bRet = getDevices(NdisPhysicalMediumNative802_11, vectRow); //WLAN adapters
    if (bRet)
    {
        for (auto it = vectRow.begin(); it != vectRow.end(); it++)
        {
            if (isNetIFConnected(*it, IF_TYPE_IEEE80211))
            {
                return L"WIFI";
            }
        }
    }

    return L"Unknown";
}

感谢@specializt