2016-09-19 187 views
1

我该如何去寻找C++中用户的外部IP?我需要一种适用于任何系统的方法,而不仅仅是我的方法。此外,该系统可能位于路由器后面,因此NAT将发挥作用,使得难以检索外部IP。如何在C++中获取外部IP地址?

理想情况下,我想这样做,而不使用任何第三方服务,如whatsmyip。但是,我不确定这是否可能。如果我不使用第三方服务,我必须通过路由器,如果ping被禁用,我猜这可能是不可能的(我可能是错的,不太确定)。

如果我要使用像whatsmyip这样的第三方服务,我该如何解决这个问题?是否有他们暴露的Web服务?我已经看到这个链接:http://automation.whatismyip.com/n09230945.asp但它似乎并没有工作。 是否有可能通过使用某些HTTP方法获取外部IP并检索它,或者我是否需要刮取页面以从中获取IP?

我仅限于使用Windows API来实现这一点(没有第三方的API)

+2

你不知道。我的意思是你没有使用C++获取接口的IP地址(* any * interface),因为C++没有任何类型的网络概念。而是使用本机平台特定的功能来获取接口列表。如果你搜索一下,你会在互联网上找到很多关于如何做到这一点的例子,包括很多在这个网站上,尝试搜索例如*枚举网络接口窗口api * –

+0

@JoachimPileborg - 我知道如何枚举网络接口。但是,我特别需要使用C++编程方式查找外部IP。我希望有本地API可以帮助我完成这一点。 – spdcbr

+0

你是指你的电脑“外部”地址,还是路由器?你可以很容易地找出一个,但不是其他的。可以使用http://www.whatsmyip.org/之类的东西,也可以直接查找路由器。如果您不想使用第三方http请求库,请查看[Windows Internet函数](https://msdn.microsoft.com/en-us/library/windows/desktop/aa383630(v = vs.85)的.aspx)。 –

回答

1

这有什么做与C++或任何特别的语言。
此外,知道你的路由器的地址不会对你有什么好处,你会知道它是NAT地址,而不是外部地址,也可能你的路由器不是通往互联网的最后一个路由器。
您必须使用外部实体 - 通常是某种可以报告与之联系的路由器的IP地址的Web服务。
以下是有人询问如何在PHP中获取客户端IP地址的sample question

+0

我不同意。他的问题是一个可以解决的问题,只要你愿意使用HTTP GET(任何其他网站会告诉你你是外部IP地址)。你如何做到这一点在Windows C++是不同的,然后C#或Python。所以我认为它对Windows C++特有的问题和答案很有帮助。感谢Software_Designer提供了一个可行的答案。 –

0

您可以使用类似curl或curlpp的库来获取http://myexternalip.com/raw的内容。它的回应只是你的外部IP,你可以阅读和使用它。

+0

这是第三方图书馆,我猜。有没有我可以完成同样的事情的窗口库? – spdcbr

+0

如果您使用的是Unix,我建议您使用exec并执行“wget http://myexternalip.com/raw”命令将内容下载为html,然后阅读该文件。我不知道是否有命令在Windows下载html内容。如果有的话,你可以从你的应用程序执行它,然后读取下载的html文件。 –

1

enter image description here

这里是winsock的方式。它只是提取嵌入在服务器发送的HTML代码中的IP地址。

此代码使用http://api.ipify.org/

下面是两个不同的代码。一个用于VS2012-2015,使用strcpy_s(),另一个用于使用strcpy()的Visual C++ 6.0,因为VS2012-2015会抛出错误消息,提示您使用strcpy_s()而不是strcpy()

Visual Studio 2012-2015的代码。

#include "stdafx.h" 
#include <string.h> 
#include <winsock2.h> 
#include <windows.h> 
#include <iostream> 
#include <vector> 
#include <locale> 
#include <sstream> 
using namespace std; 
#pragma comment(lib,"ws2_32.lib") 


string website_HTML; 
locale local; 
void get_Website(string url); 
char lineBuffer[200][80] = { ' ' }; 
char buffer[10000]; 
char ip_address[16]; 
int i = 0, bufLen = 0, j = 0, lineCount = 0; 
int lineIndex = 0, posIndex = 0; 

//**************************************************** 

int main(void){ 
    cout << "\n\n\n"; 
    get_Website("api.ipify.org"); 
    for (size_t i = 0; i<website_HTML.length(); ++i) website_HTML[i] = tolower(website_HTML[i], local); 

    istringstream ss(website_HTML); 
    string stoken; 

    while (getline(ss, stoken, '\n')) { 

     //cout <<"-->"<< stoken.c_str() << '\n'; 

     strcpy_s(lineBuffer[lineIndex], stoken.c_str()); 
     int dot = 0; 
     for (int ii = 0; ii< strlen(lineBuffer[lineIndex]); ii++){ 

      if (lineBuffer[lineIndex][ii] == '.') dot++; 
      if (dot >= 3){ 
       dot = 0; 
       strcpy_s(ip_address, lineBuffer[lineIndex]); 
      } 
     } 

     lineIndex++; 
    } 
    cout << "Your IP Address is " << ip_address << " \n\n"; 


    cout << "\nPress ANY key to close.\n\n"; 
    cin.ignore(); cin.get(); 

    return 0; 
} 

//**************************************************** 

void get_Website(string url){ 
    WSADATA wsaData; 
    SOCKET Socket; 
    SOCKADDR_IN SockAddr; 
    int lineCount = 0; 
    int rowCount = 0; 
    struct hostent *host; 
    string get_http; 


    get_http = "GET/HTTP/1.1\r\nHost: " + url + "\r\nConnection: close\r\n\r\n"; 

    if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0){ 
     cout << "WSAStartup failed.\n"; 
     system("pause"); 
     //return 1; 
    } 

    Socket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); 
    host = gethostbyname(url.c_str()); 

    SockAddr.sin_port = htons(80); 
    SockAddr.sin_family = AF_INET; 
    SockAddr.sin_addr.s_addr = *((unsigned long*)host->h_addr); 

    if (connect(Socket, (SOCKADDR*)(&SockAddr), sizeof(SockAddr)) != 0){ 
     cout << "Could not connect"; 
     system("pause"); 
     //return 1; 
    } 
    send(Socket, get_http.c_str(), strlen(get_http.c_str()), 0); 

    int nDataLength; 
    while ((nDataLength = recv(Socket, buffer, 10000, 0)) > 0){ 
     int i = 0; 
     while (buffer[i] >= 32 || buffer[i] == '\n' || buffer[i] == '\r'){ 

      website_HTML += buffer[i]; 
      i += 1; 
     } 
    } 

    closesocket(Socket); 
    WSACleanup(); 

} 

的Visual C++ 6.0代码

#include <string.h> 
    #include <winsock2.h> 
    #include <windows.h> 
    #include <iostream> 
    #include <vector> 
    #include <locale> 
    #include <sstream> 
    using namespace std; 
    #pragma comment(lib,"ws2_32.lib") 


    string website_HTML; 
    locale local; 
    void get_Website(string url); 
    char lineBuffer[200][80] ={' '}; 
    char buffer[10000]; 
    char ip_address[16]; 
    int i = 0, bufLen=0, j=0,lineCount=0; 
    int lineIndex=0, posIndex=0; 

    //**************************************************** 

    int main(void){ 
     cout << "\n\n\n"; 
     get_Website("api.ipify.org"); 
     for (size_t i=0; i<website_HTML.length(); ++i) website_HTML[i]= tolower(website_HTML[i],local); 

     istringstream ss(website_HTML); 
     string stoken; 

     while(getline(ss, stoken, '\n')) { 

        strcpy(lineBuffer[lineIndex],stoken.c_str()); 
        int dot=0; 
        for (int ii=0; ii< strlen(lineBuffer[lineIndex]); ii++){ 

         if (lineBuffer[lineIndex][ii] == '.') dot++; 
         if (dot>=3){ 
          dot=0; 
          strcpy(ip_address,lineBuffer[lineIndex]); 
         } 
        } 
        lineIndex++; 
     } 
     cout<<"Your IP Address is "<< ip_address<<" \n\n"; 

     cout<<"\nPress ANY key to close.\n\n"; 
     cin.ignore(); cin.get(); 


    return 0; 
} 

//**************************************************** 

void get_Website(string url){ 
    WSADATA wsaData; 
    SOCKET Socket; 
    SOCKADDR_IN SockAddr; 
    int lineCount=0; 
    int rowCount=0; 
    struct hostent *host; 
    string get_http; 


    get_http = "GET/HTTP/1.1\r\nHost: " + url + "\r\nConnection: close\r\n\r\n"; 

    if (WSAStartup(MAKEWORD(2,2), &wsaData) != 0){ 
     cout << "WSAStartup failed.\n"; 
     system("pause"); 
     //return 1; 
    } 

    Socket=socket(AF_INET,SOCK_STREAM,IPPROTO_TCP); 
    host = gethostbyname(url.c_str()); 

    SockAddr.sin_port=htons(80); 
    SockAddr.sin_family=AF_INET; 
    SockAddr.sin_addr.s_addr = *((unsigned long*)host->h_addr); 

    if(connect(Socket,(SOCKADDR*)(&SockAddr),sizeof(SockAddr)) != 0){ 
     cout << "Could not connect"; 
     system("pause"); 
     //return 1; 
    } 
    send(Socket,get_http.c_str(), strlen(get_http.c_str()),0); 

    int nDataLength; 
    while ((nDataLength = recv(Socket,buffer,10000,0)) > 0){   
     int i = 0; 
     while (buffer[i] >= 32 || buffer[i] == '\n' || buffer[i] == '\r'){ 

      website_HTML+=buffer[i]; 
      i += 1; 
     }    
    } 

    closesocket(Socket); 
    WSACleanup(); 

} 
1

在理论上,你可以使用Windows的UPnP API来做到这一点。您首先要使用UPnPDeviceFinder来枚举Internet网关设备。然后你会得到一个IUPnPRemoteEndpointInfo为网关(当然,还有通常只有一个,反正),并调用它GetStringValue,使含"RemoteAddress"一个字符串来得到它的远程地址(我认为意味着它的外部地址,但我会承认我不完全确定)。哦,因为这是COM,那必须是一个系统字符串,而不是一个普通的字符串。

从外部提供商获取IP是lot更容易。不使用任何3个党库,它的代码如下所示:

#include <windows.h> 
#include <wininet.h> 
#include <string> 
#include <iostream> 

std::string real_ip() { 

    HINTERNET net = InternetOpen("IP retriever", 
     INTERNET_OPEN_TYPE_PRECONFIG, 
     NULL, 
     NULL, 
     0); 

    HINTERNET conn = InternetOpenUrl(net, 
            "http://myexternalip.com/raw", 
             NULL, 
             0, 
             INTERNET_FLAG_RELOAD, 
             0); 

    char buffer[4096]; 
    DWORD read; 

    InternetReadFile(conn, buffer, sizeof(buffer)/sizeof(buffer[0]), &read); 
    InternetCloseHandle(net);  

    return std::string(buffer, read); 
} 

int main() { 
    std::cout << real_ip() << "\n"; 
} 

编译:

cl ip_addr.cpp wininet.lib 

注意:如果您的机器配置为使用IPv6,这样可以(并且将)检索你的IPv6地址)。