如何将字符串转换为 IP 地址,反之亦然

Posted

技术标签:

【中文标题】如何将字符串转换为 IP 地址,反之亦然【英文标题】:How to convert string to IP address and vice versa 【发布时间】:2022-01-13 09:34:14 【问题描述】:

如何转换字符串 ipAddress (struct in_addr),反之亦然? 以及如何上交无符号长 IP 地址? 谢谢

【问题讨论】:

WSAAddressToStringWSAStringToAddress 【参考方案1】:

如果您需要其他方式,请使用 inet_ntop()inet_pton()。不要使用 inet_ntoa(), inet_aton() 和类似的,因为它们已被弃用并且不支持 ipv6。

这是一个不错的guide,有很多例子。

// IPv4 demo of inet_ntop() and inet_pton()

struct sockaddr_in sa;
char str[INET_ADDRSTRLEN];

// store this IP address in sa:
inet_pton(AF_INET, "192.0.2.33", &(sa.sin_addr));

// now get it back and print it
inet_ntop(AF_INET, &(sa.sin_addr), str, INET_ADDRSTRLEN);

printf("%s\n", str); // prints "192.0.2.33"

【讨论】:

在Windows下我应该怎么做才能使用它们?我应该包括什么? 如果在 windows vista 及更高版本上使用 winsock,则使用 InetNtop 和 InetPton。头文件 Ws2tcpip.h 。提示是查看 msdn 上的示例。 Winsock2.h 中的 WSAAddressToString (ntop) 和 WSAStringToAddress (pton)。我建议至少升级到 vista,并问你下次提问时指定你正在开发的平台:) 此链接不再可用 @QiFan “n”代表“网络”,“p”代表“(文本)演示”。【参考方案2】:

我不确定我是否正确理解了这个问题。

不管怎样,你在找这个吗:

std::string ip ="192.168.1.54";
std::stringstream s(ip);
int a,b,c,d; //to store the 4 ints
char ch; //to temporarily store the '.'
s >> a >> ch >> b >> ch >> c >> ch >> d;
std::cout << a << "  " << b << "  " << c << "  "<< d;

Output:

192  168  1  54

【讨论】:

【参考方案3】:

我能够使用以下代码将字符串转换为 DWORD 并返回:

char strAddr[] = "127.0.0.1"
DWORD ip = inet_addr(strAddr); // ip contains 16777343 [0x0100007f in hex]

struct in_addr paddr;
paddr.S_un.S_addr = ip;

char *strAdd2 = inet_ntoa(paddr); // strAdd2 contains the same string as strAdd

我正在从事旧 MFC 代码的维护项目,因此转换不推荐使用的函数调用不适用。

【讨论】:

【参考方案4】:

inet_ntoa()in_addr 转换为字符串:

inet_ntoa 函数将一个 (ipv4) Internet网络地址成 Internet 标准中的 ASCII 字符串 点分十进制格式。

inet_addr() 做相反的工作

inet_addr 函数将一个 包含 IPv4 的字符串 点分十进制地址转换为适当的 IN_ADDR 结构的地址

PS 这是谷歌搜索“in_addr to string”的第一个结果!

【讨论】:

这些功能已被弃用,不应使用。 @Milan 好的。虽然 OP 使用处理 ipv4 的 in_addr 结构 在Windows下我应该怎么做才能使用它们?我应该包括什么? 答案在函数的文档中,(点击链接):包括 Winsock2.h。【参考方案5】:

将字符串转换为in-addr:

in_addr maskAddr;
inet_aton(netMaskStr, &maskAddr);

将 in_addr 转换为字符串:

char saddr[INET_ADDRSTRLEN];
inet_ntop(AF_INET, &inaddr, saddr, INET_ADDRSTRLEN);

【讨论】:

【参考方案6】:

这个例子展示了如何从字符串转换为ip,反之亦然:

struct sockaddr_in sa;
char ip_saver[INET_ADDRSTRLEN];

// store this IP address in sa:
inet_pton(AF_INET, "192.0.1.10", &(sa.sin_addr));

// now get it back 
sprintf(ip_saver, "%s", sa.sin_addr));

// prints "192.0.2.10"
printf("%s\n", ip_saver); 

【讨论】:

添加标题也不错【参考方案7】:

这里有易于使用、线程安全的 c++ 函数,用于将 uint32_t native-endian 转换为 string,并将 string 转换为 native-endian uint32_t:

#include <arpa/inet.h> // inet_ntop & inet_pton
#include <string.h> // strerror_r
#include <arpa/inet.h> // ntohl & htonl
using namespace std; // im lazy

string ipv4_int_to_string(uint32_t in, bool *const success = nullptr)

    string ret(INET_ADDRSTRLEN, '\0');
    in = htonl(in);
    const bool _success = (NULL != inet_ntop(AF_INET, &in, &ret[0], ret.size()));
    if (success)
    
        *success = _success;
    
    if (_success)
    
        ret.pop_back(); // remove null-terminator required by inet_ntop
    
    else if (!success)
    
        char buf[200] = 0;
        strerror_r(errno, buf, sizeof(buf));
        throw std::runtime_error(string("error converting ipv4 int to string ") + to_string(errno) + string(": ") + string(buf));
    
    return ret;

// return is native-endian
// when an error occurs: if success ptr is given, it's set to false, otherwise a std::runtime_error is thrown.
uint32_t ipv4_string_to_int(const string &in, bool *const success = nullptr)

    uint32_t ret;
    const bool _success = (1 == inet_pton(AF_INET, in.c_str(), &ret));
    ret = ntohl(ret);
    if (success)
    
        *success = _success;
    
    else if (!_success)
    
        char buf[200] = 0;
        strerror_r(errno, buf, sizeof(buf));
        throw std::runtime_error(string("error converting ipv4 string to int ") + to_string(errno) + string(": ") + string(buf));
    
    return ret;

公平警告,截至发稿时,它们未经测试。但这些功能正是我来这个帖子时所寻找的。​​p>

【讨论】:

【参考方案8】:

第三个inet_pton 参数是一个指向in_addr 结构的指针。成功调用inet_pton 后,in_addr 结构将填充地址信息。该结构的S_addr 字段包含网络字节顺序(倒序)的IP 地址。

Example : 

#include <arpa/inet.h>
uint32_t NodeIpAddress::getIPv4AddressInteger(std::string IPv4Address) 
    int result;
    uint32_t IPv4Identifier = 0;
    struct in_addr addr;
    // store this IP address in sa:
    result = inet_pton(AF_INET, IPv4Address.c_str(), &(addr));
    if (result == -1)          
gpLogFile->Write(LOGPREFIX, LogFile::LOGLEVEL_ERROR, _T("Failed to convert IP %hs to IPv4 Address. Due to invalid family of %d. WSA Error of %d"), IPv4Address.c_str(), AF_INET, result);
    
    else if (result == 0) 
        gpLogFile->Write(LOGPREFIX, LogFile::LOGLEVEL_ERROR, _T("Failed to convert IP %hs to IPv4"), IPv4Address.c_str());
    
    else 
        IPv4Identifier = ntohl(*((uint32_t *)&(addr)));
    
    return IPv4Identifier;

【讨论】:

【参考方案9】:

十六进制IP地址转字符串IP

#include <iostream>
#include <sstream>
using namespace std;

int main()

    uint32_t ip = 0x0AA40001;
    string ip_str="";
    int temp = 0;
    for (int i = 0; i < 8; i++)
        if (i % 2 == 0)
        
            temp += ip & 15;
            ip = ip >> 4;
        
        else
        
            stringstream ss;
            temp += (ip & 15) * 16;
            ip = ip >> 4;
            ss << temp;
            ip_str = ss.str()+"." + ip_str;
            temp = 0;
        
    
    ip_str.pop_back();
    cout << ip_str;

输出:10.164.0.1

【讨论】:

以上是关于如何将字符串转换为 IP 地址,反之亦然的主要内容,如果未能解决你的问题,请参考以下文章

用C++如何将一个整数转换成一个IP地址?

将十六进制字符串转换为 IP 地址

如何将IP地址项转换为字符串c#

如何使用 boost::asio 将 URL 转换为 IP 地址?

如何将字符串转换为十六进制,反之亦然?

如何将标准 IP 地址格式字符串转换为十六进制和长字符串?