iPhoneiPadOSX:如何以编程方式获取IP地址?

iPhoneiPadOSX:如何以编程方式获取IP地址?,第1张

概述我想以编程方式获取我的iPad的IP地址。 如何查询网络子系统以了解我的IPv4(和IPv6)地址是什么? 谢谢。 PS:我可以禁用IPv6吗? 以下代码在iOS或OSX设备上查找所有IPv4和IPv6地址。第一个getIPAddress方法或多或少地作为这个答案中的旧代码:你可以喜欢一个或另一个类型的地址,它总是喜欢WIFI通过蜂窝(显然你可以改变这个)。 更有趣的是,它可以返回找到的所有地址的 我想以编程方式获取我的iPad的IP地址。
如何查询网络子系统以了解我的IPv4(和IPv6)地址是什么?

谢谢。
PS:我可以禁用IPv6吗?

解决方法 以下代码在iOS或OSX设备上查找所有IPv4和IPv6地址。第一个getIPAddress方法或多或少地作为这个答案中的旧代码:你可以喜欢一个或另一个类型的地址,它总是喜欢WIFI通过蜂窝(显然你可以改变这个)。

更有趣的是,它可以返回找到的所有地址的字典,跳过未上传接口的地址或与环回相关联的地址。前面的代码以及关于这个主题的其他解决方案不会正确地解码IPv6(inet_ntoa不能处理它们)。这是在Jens Alfke在Apple论坛上指出的 – 使用的正确功能是inet_ntop(看看手册页,或者参考Jens提供的inet_ntop这篇文章。

字典键具有“interface”“/”“ipv4或ipv6”的形式。

#include <ifaddrs.h>#include <arpa/inet.h>#include <net/if.h>#define IOS_CELLulAR    @"pdp_ip0"#define IOS_WIFI        @"en0"//#define IOS_VPN       @"utun0"#define IP_ADDR_IPv4    @"ipv4"#define IP_ADDR_IPv6    @"ipv6"- (Nsstring *)getIPAddress:(BOol)preferIPv4{    NSArray *searchArray = preferIPv4 ?                            @[ /*IOS_VPN @"/" IP_ADDR_IPv4,IOS_VPN @"/" IP_ADDR_IPv6,*/ IOS_WIFI @"/" IP_ADDR_IPv4,IOS_WIFI @"/" IP_ADDR_IPv6,IOS_CELLulAR @"/" IP_ADDR_IPv4,IOS_CELLulAR @"/" IP_ADDR_IPv6 ] :                            @[ /*IOS_VPN @"/" IP_ADDR_IPv6,IOS_VPN @"/" IP_ADDR_IPv4,*/ IOS_WIFI @"/" IP_ADDR_IPv6,IOS_WIFI @"/" IP_ADDR_IPv4,IOS_CELLulAR @"/" IP_ADDR_IPv6,IOS_CELLulAR @"/" IP_ADDR_IPv4 ] ;    NSDictionary *addresses = [self getIPAddresses];    NSLog(@"addresses: %@",addresses);    __block Nsstring *address;    [searchArray enumerateObjectsUsingBlock:^(Nsstring *key,NSUInteger IDx,BOol *stop)        {            address = addresses[key];            if(address) *stop = YES;        } ];    return address ? address : @"0.0.0.0";}- (NSDictionary *)getIPAddresses{    NSMutableDictionary *addresses = [NSMutableDictionary dictionaryWithCapacity:8];    // 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,INET6_ADDRSTRLEN)) {                        type = IP_ADDR_IPv6;                    }                }                if(type) {                    Nsstring *key = [Nsstring stringWithFormat:@"%@/%@",name,type];                    addresses[key] = [Nsstring stringWithUTF8String:addrBuf];                }            }        }        // Free memory        freeifaddrs(interfaces);    }    return [addresses count] ? addresses : nil;}

EDIT1:代码更新于2014年5月16日(错误由lhunath指出,见注释)。环回地址现在返回,但它很容易你取消注释测试以自己排除它们。

编辑2:(由一些未知的人):进一步改进2015年3月13日:如果用户使用VPN(不管是通过WiFi还是蜂窝),以前的代码将失败。现在,它甚至与VPN连接工作。 VPN连接优先于WiFi和Cell,因为这是设备处理它的方式。这甚至适用于Macs,因为在Mac上的VPN连接也使用IF utun0但未测试。

EDIT3:(9/8/2016)鉴于@Qiulang遇到的问题(见评论)与VPN代码(别人添加),我已经评论过了。如果任何人明确地知道如何指定用户VPN,请提出评论。

总结

以上是内存溢出为你收集整理的iPhone/iPad/OSX:如何以编程方式获取IP地址?全部内容,希望文章能够帮你解决iPhone/iPad/OSX:如何以编程方式获取IP地址?所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/web/1088220.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-05-27
下一篇 2022-05-27

发表评论

登录后才能评论

评论列表(0条)

保存