使用 Objective-C 获取 Mac 的所有 IP 地址

Get All of Mac's IP Addresses Using Objective-C

我正在尝试获取我的 Mac 的所有 IP 地址 (IPv4 + IPv6),但没有看到我预期的结果。我正在使用 this Stack Overflow post 中的代码来获取所有 IP 地址的数组,但是缺少一个 IP 地址。

我正在使用另一个 Mac 来共享它的互联网连接并根据 Apple's documentation for testing/supporting IPv6 networks 创建一个 NAT64 网络。

例如,系统偏好设置 → 网络窗格显示我的 IP 地址是:

...但我发现经过进一步检查我实际上有两个 IPv6 地址:

...但只返回其中一个:

"169.254.38.213",
"2001:2:0:aab1:d0ef:646d:f22a:5d83",
"127.0.0.1"

...使用时:

#include <ifaddrs.h>
#include <arpa/inet.h>
#include <net/if.h>

@interface AppDelegate ()

#define IP_ADDR_IPv4    @"ipv4"
#define IP_ADDR_IPv6    @"ipv6"

@end

@implementation AppDelegate

- (void) applicationDidFinishLaunching: (NSNotification *) aNotification
{
    NSMutableDictionary *addresses = [NSMutableDictionary dictionaryWithCapacity:8];
    NSMutableArray *ipAddressesArray = [[NSMutableArray alloc] init];

    // retrieve the current interfaces - returns 0 on success
    struct ifaddrs *interfaces;

    if (!getifaddrs(&interfaces))
    {
        // Loop through linked list of interfaces
        struct ifaddrs *interface;

        for (interface=interfaces; interface; interface=interface->ifa_next)
        {
            if (!(interface->ifa_flags & IFF_UP) /* || (interface->ifa_flags & IFF_LOOPBACK) */)
            {
                continue; // deeply nested code harder to read
            }

            const struct sockaddr_in *addr = (const struct sockaddr_in*)interface->ifa_addr;
            char addrBuf[ MAX(INET_ADDRSTRLEN, INET6_ADDRSTRLEN) ];

            if (addr && (addr->sin_family==AF_INET || addr->sin_family==AF_INET6))
            {
                NSString *name = [NSString stringWithUTF8String:interface->ifa_name];
                NSString *type;

                if (addr->sin_family == AF_INET)
                {
                    if (inet_ntop(AF_INET, &addr->sin_addr, addrBuf, INET_ADDRSTRLEN))
                    {
                        type = IP_ADDR_IPv4;
                    }
                }
                else
                {
                    const struct sockaddr_in6 *addr6 = (const struct sockaddr_in6*)interface->ifa_addr;

                    if (inet_ntop(AF_INET6, &addr6->sin6_addr, addrBuf, INET6_ADDRSTRLEN))
                    {
                        type = IP_ADDR_IPv6;
                    }
                }

                if (type)
                {
                    NSString *key = [NSString stringWithFormat:@"%@/%@", name, type];
                    addresses[key] = [NSString stringWithUTF8String:addrBuf];
                }
            }
        }

        for (id key in addresses)
        {
            if (![[addresses valueForKey:key] hasPrefix:@"fe80"])
            {
                [ipAddressesArray addObject:[addresses valueForKey:key]];
            }
        }

        // Free memory
        freeifaddrs(interfaces);
    }

    NSLog(@"%@", ipAddressesArray);
}

知道这里发生了什么吗?有关系吗?我正在尝试根据 IP 地址匹配有条件地执行一些其他代码。如果返回的唯一 IPv6 地址是第一次打开网络窗格时在系统偏好设置中向用户显示的地址,那将是一回事,但返回的唯一 IPv6 地址是您必须深入研究的 "hidden" 地址高级...部分找到。提前谢谢你。

这里的答案是我是个白痴。正如 Rob Napier 在他们对我的问题的评论中所建议的那样,我实际上是在过滤掉丢失的 IP 地址:

NSString *key = [NSString stringWithFormat:@"%@/%@", name, type];
addresses[key] = [NSString stringWithUTF8String:addrBuf];

每个接口可以有多个 IP 地址,但由于我在 addresses 字典中使用接口类型作为唯一键,字典中每个接口只有一个 IP 地址。我通过返回字典数组而不是单个字典来修复此问题:

if (![[NSString stringWithFormat:@"%s", addrBuf] hasPrefix:@"fe80"])
{
      NSDictionary *addressDict = [NSDictionary dictionaryWithObjectsAndKeys :
                                                   [NSString stringWithFormat:@"%@/%@", name, type], @"Interface",
                                                   [NSString stringWithFormat:@"%s", addrBuf], @"Address",
                                                    nil];

     [addresses addObject:addressDict];
}

return addresses;