如何以编程方式获取iPhone的MAC地址


144

如何以编程方式获取iPhone的MAC地址和IP地址?


我想在iPhone应用程序中以编程方式获取MAC地址和IP地址。

Wifi MAC地址?还是蓝牙?有很多Mac地址。
mskw

11
从iOS 7开始,02:00:00:00:00:00当您要求任何设备上的MAC地址时,系统总是返回该值。在下面检查我的答案。
Hejazi 2013年

对于ios及更高版本(直到今天),请参考stackoverflow.com/questions/11836225/…–
rptwsthi 2013年

Answers:


114

注意从iOS7开始,您将无法再检索设备MAC地址。将返回固定值,而不是实际的MAC


不久前,我偶然跌跌撞撞。最初,我从这里进行了一些修改,并清理了所有内容。

IPAddress.h
IPAddress.c

并使用它

InitAddresses();
GetIPAddresses();
GetHWAddresses();

int i;
NSString *deviceIP = nil;
for (i=0; i<MAXADDRS; ++i)
{
    static unsigned long localHost = 0x7F000001;        // 127.0.0.1
    unsigned long theAddr;

    theAddr = ip_addrs[i];

    if (theAddr == 0) break;
    if (theAddr == localHost) continue;

    NSLog(@"Name: %s MAC: %s IP: %s\n", if_names[i], hw_addrs[i], ip_names[i]);

        //decided what adapter you want details for
    if (strncmp(if_names[i], "en", 2) == 0)
    {
        NSLog(@"Adapter en has a IP of %s", ip_names[i]);
    }
}

适配器名称因模拟器/设备以及设备上的wifi或小区而异。


我只是想知道这是否与通用OS X方法有什么不同(我问是因为我是mac n00b)
Tamas Czinege 09年

2
@abc:我建议您发布有关此问题的另一个问题。最好将其掩埋在其他问题的评论中。
PyjamaSam,2009年

1
但是对ether_ntoa的调用呢?这是私有API吗?
stigi

1
@NicTesla我不能肯定地发表评论,但是由于它只是在进行基本的网络通话,所以我看不到任何问题。尽管从iOS5中删除UDID可以说是这样,但苹果在将来的某个时候可能会仔细检查mac地址并使用它来开发某种替代的唯一标识符。
PyjamaSam 2011年

1
关于ether_ntoa警告,请查看:stackoverflow.com/a/13412265/88597
ohho,2012年

45

更新:这在iOS 7上不起作用。您应该使用ASIdentifierManager


MobileDeveloperTips网站上的更干净的解决方案

#include <sys/socket.h>
#include <sys/sysctl.h>
#include <net/if.h>
#include <net/if_dl.h>

...

- (NSString *)getMacAddress
{
  int                 mgmtInfoBase[6];
  char                *msgBuffer = NULL;
  size_t              length;
  unsigned char       macAddress[6];
  struct if_msghdr    *interfaceMsgStruct;
  struct sockaddr_dl  *socketStruct;
  NSString            *errorFlag = NULL;

  // Setup the management Information Base (mib)
  mgmtInfoBase[0] = CTL_NET;        // Request network subsystem
  mgmtInfoBase[1] = AF_ROUTE;       // Routing table info
  mgmtInfoBase[2] = 0;              
  mgmtInfoBase[3] = AF_LINK;        // Request link layer information
  mgmtInfoBase[4] = NET_RT_IFLIST;  // Request all configured interfaces

  // With all configured interfaces requested, get handle index
  if ((mgmtInfoBase[5] = if_nametoindex("en0")) == 0) 
    errorFlag = @"if_nametoindex failure";
  else
  {
    // Get the size of the data available (store in len)
    if (sysctl(mgmtInfoBase, 6, NULL, &length, NULL, 0) < 0) 
      errorFlag = @"sysctl mgmtInfoBase failure";
    else
    {
      // Alloc memory based on above call
      if ((msgBuffer = malloc(length)) == NULL)
        errorFlag = @"buffer allocation failure";
      else
      {
        // Get system information, store in buffer
        if (sysctl(mgmtInfoBase, 6, msgBuffer, &length, NULL, 0) < 0)
          errorFlag = @"sysctl msgBuffer failure";
      }
    }
  }

  // Befor going any further...
  if (errorFlag != NULL)
  {
    NSLog(@"Error: %@", errorFlag);
    return errorFlag;
  }

  // Map msgbuffer to interface message structure
  interfaceMsgStruct = (struct if_msghdr *) msgBuffer;

  // Map to link-level socket structure
  socketStruct = (struct sockaddr_dl *) (interfaceMsgStruct + 1);

  // Copy link layer address data in socket structure to an array
  memcpy(&macAddress, socketStruct->sdl_data + socketStruct->sdl_nlen, 6);

  // Read from char array into a string object, into traditional Mac address format
  NSString *macAddressString = [NSString stringWithFormat:@"%02X:%02X:%02X:%02X:%02X:%02X", 
                                macAddress[0], macAddress[1], macAddress[2], 
                                macAddress[3], macAddress[4], macAddress[5]];
  NSLog(@"Mac Address: %@", macAddressString);

  // Release the buffer memory
  free(msgBuffer);

  return macAddressString;
}

将链接更改为mobiledevelopertips而不是iphonedevelopertips
SEG

苹果很可能会在iOS7中打破这个问题……除非正式发布,否则我们将无法对此发表评论,除非是在官方的Apple Dev论坛上。
加布

3
这在装有iOS7的iPhone5上不起作用。它给我:02:00:00:00:00:00这是不对的。
Sjoerd Perfors

3
@Brenden您再也无法在iOS7上使用PublicAPI获取网络MAC地址。因此,如果您需要唯一的ID,则应使用IdentifierManager
ArtFeel 2013年

2
@SjoerdPerfors在iOS 7下,您确实会获得与02:00:00:00:00:00苹果公司相同的地址,作为隐私保护措施。
罗勒·布尔克

31

无论是否启用了wifi,我都想返回一个地址,所以选择的解决方案对我不起作用。在进行一些调整后,我使用了在某个论坛上找到的另一个电话。我得到以下结果(对我生锈的C表示抱歉):

#include <sys/types.h>
#include <stdio.h>
#include <string.h>
#include <sys/socket.h>
#include <net/if_dl.h>
#include <ifaddrs.h>


char*  getMacAddress(char* macAddress, char* ifName) {

int  success;
struct ifaddrs * addrs;
struct ifaddrs * cursor;
const struct sockaddr_dl * dlAddr;
const unsigned char* base;
int i;

success = getifaddrs(&addrs) == 0;
if (success) {
    cursor = addrs;
    while (cursor != 0) {
        if ( (cursor->ifa_addr->sa_family == AF_LINK)
            && (((const struct sockaddr_dl *) cursor->ifa_addr)->sdl_type == IFT_ETHER) && strcmp(ifName,  cursor->ifa_name)==0 ) {
            dlAddr = (const struct sockaddr_dl *) cursor->ifa_addr;
            base = (const unsigned char*) &dlAddr->sdl_data[dlAddr->sdl_nlen];
            strcpy(macAddress, ""); 
            for (i = 0; i < dlAddr->sdl_alen; i++) {
                if (i != 0) {
                    strcat(macAddress, ":");
                }
                char partialAddr[3];
                sprintf(partialAddr, "%02X", base[i]);
                strcat(macAddress, partialAddr);

            }
        }
        cursor = cursor->ifa_next;
    }

    freeifaddrs(addrs);
}    
return macAddress;
}

然后我将其称为en0,如下所示:

char* macAddressString= (char*)malloc(18);
NSString* macAddress= [[NSString alloc] initWithCString:getMacAddress(macAddressString, "en0")
                                              encoding:NSMacOSRomanStringEncoding];
free(macAddressString);

9
我必须添加#define IFT_ETHER 0x6
teedyay'8

这对我不起作用...说ifName和macaddress变量一样没有声明。
bugfixr 2011年

上面的调用代码将泄漏。免费(macAddressString); 从上述调用方法返回之前。另外,如果上述调用代码在另一种方法中,则应自动释放macAddress。

4
需要明确的是,代码本身不会泄漏,评论者在最后讨论了调用代码片段,在这里我分配了两个字符串而不释放它们。
船长

1
我将其放入一个不错的Objective-C方法中并使其不需要参数:gist.github.com/wsidell/5069159
wsidell 2013年

23

从iOS 7开始,02:00:00:00:00:00当您要求任何设备上的MAC地址时,系统总是返回该值。

在iOS 7及更高版本中,如果您要求提供iOS设备的MAC地址,系统将返回值02:00:00:00:00:00。如果需要标识设备,请改用UIDevice的identifierForVendor属性。(为自己的广告目的而需要标识符的应用应考虑改用ASIdentifierManager的advertisingIdentifier属性。)”

参考:阅读发布


API中这种弃用的真正好处在于,它仍然可以在模拟器上正确运行,并且目前仅在硬件上运行。
John Nye 2013年

12

有各种各样的解决方案,但是我找不到完整的东西。因此,我针对以下问题制定了自己的解决方案:

资讯网

如何使用 :

NICInfoSummary* summary = [[[NICInfoSummary alloc] init] autorelease];

// en0 is for WiFi 
NICInfo* wifi_info = [summary findNICInfo:@"en0"];

// you can get mac address in 'XX-XX-XX-XX-XX-XX' form
NSString* mac_address = [wifi_info getMacAddressWithSeparator:@"-"];

// ip can be multiple
if(wifi_info.nicIPInfos.count > 0)
{
    NICIPInfo* ip_info = [wifi_info.nicIPInfos objectAtIndex:0];
    NSString* ip = ip_info.ip;
    NSString* netmask = ip_info.netmask;
    NSString* broadcast_ip = ip_info.broadcastIP;
}
else
{
    NSLog(@"WiFi not connected!");
}

非常好和有用的课程!干得好!但是,您在两个类上都错过了[super dealloc],却忘记了包含一个库,这会导致xCode中出现警告。修补程序版本可以在这里找到:raptor.hk/download/NICInfo_raptor_patched.zip
Raptor

得到此结果的Mac地址:02:00:00:00:00:00
斯大林·普斯帕拉杰

1
是。苹果这个封闭的iOS以来7 :( ... developer.apple.com/library/content/releasenotes/General/...
Kenial

5

这看起来像一个非常干净的解决方案:UIDevice BIdentifier

// Return the local MAC addy
// Courtesy of FreeBSD hackers email list
// Accidentally munged during previous update. Fixed thanks to erica sadun & mlamb.
- (NSString *) macaddress{

    int                 mib[6];
    size_t              len;
    char                *buf;
    unsigned char       *ptr;
    struct if_msghdr    *ifm;
    struct sockaddr_dl  *sdl;

    mib[0] = CTL_NET;
    mib[1] = AF_ROUTE;
    mib[2] = 0;
    mib[3] = AF_LINK;
    mib[4] = NET_RT_IFLIST;

    if ((mib[5] = if_nametoindex("en0")) == 0) {
        printf("Error: if_nametoindex error\n");
        return NULL;
    }

    if (sysctl(mib, 6, NULL, &len, NULL, 0) < 0) {
        printf("Error: sysctl, take 1\n");
        return NULL;
    }

    if ((buf = malloc(len)) == NULL) {
        printf("Could not allocate memory. error!\n");
        return NULL;
    }

    if (sysctl(mib, 6, buf, &len, NULL, 0) < 0) {
        printf("Error: sysctl, take 2");
        free(buf);
        return NULL;
    }

    ifm = (struct if_msghdr *)buf;
    sdl = (struct sockaddr_dl *)(ifm + 1);
    ptr = (unsigned char *)LLADDR(sdl);
    NSString *outstring = [NSString stringWithFormat:@"%02X:%02X:%02X:%02X:%02X:%02X", 
                           *ptr, *(ptr+1), *(ptr+2), *(ptr+3), *(ptr+4), *(ptr+5)];
    free(buf);

    return outstring;
}

1
我使用您的解决方案,有99%的案例按预期运行良好,但是1%的速率无法获取mac地址,返回NULL,这是由AppStore中的用户报告的。仍然没有获得可重复的步骤,知道了吗?谢谢。
建华2012年

我还没有发现任何问题,但我会一直注意!
Grantland Chew 2012年

紧跟@jianhua的评论:我们刚收到一个基本上无法使用该应用程序的用户的报告,因为此代码为其设备返回null(我们使用标识符来验证每个服务器调用)。
samvermette 2012年

失败的案例是在具有IOS版本5.0.1的iPhone 4S上,当然仅在某些特殊情况下。
建华

5

现在,iOS 7设备始终返回02:00:00:00:00:00的MAC地址。

因此,最好使用[UIDevice identifierForVendor]。

因此最好调用此方法来获取应用程序特定的唯一密钥

类别会更合适

导入“ UIDevice + Identifier.h”

- (NSString *) identifierForVendor1
{
    if ([[UIDevice currentDevice] respondsToSelector:@selector(identifierForVendor)]) {
        return [[[UIDevice currentDevice] identifierForVendor] UUIDString];
    }
    return @"";
}

现在调用上述方法以获得唯一的地址

NSString *like_UDID=[NSString stringWithFormat:@"%@",
                [[UIDevice currentDevice] identifierForVendor1]];

NSLog(@"%@",like_UDID);

在同一应用的多次安装中,每个设备的此标识符是否一致?
samsam 2015年

4

@Grantland这个“非常干净的解决方案”看起来类似于我对iPhoneDeveloperTips解决方案的改进。

您可以在这里看到我的步骤:https : //gist.github.com/1409855/

/* Original source code courtesy John from iOSDeveloperTips.com */

#include <sys/socket.h>
#include <sys/sysctl.h>
#include <net/if.h>
#include <net/if_dl.h>

+ (NSString *)getMacAddress
{
    int                 mgmtInfoBase[6];
    char                *msgBuffer = NULL;
    NSString            *errorFlag = NULL;
    size_t              length;

    // Setup the management Information Base (mib)
    mgmtInfoBase[0] = CTL_NET;        // Request network subsystem
    mgmtInfoBase[1] = AF_ROUTE;       // Routing table info
    mgmtInfoBase[2] = 0;              
    mgmtInfoBase[3] = AF_LINK;        // Request link layer information
    mgmtInfoBase[4] = NET_RT_IFLIST;  // Request all configured interfaces

    // With all configured interfaces requested, get handle index
    if ((mgmtInfoBase[5] = if_nametoindex("en0")) == 0) 
        errorFlag = @"if_nametoindex failure";
    // Get the size of the data available (store in len)
    else if (sysctl(mgmtInfoBase, 6, NULL, &length, NULL, 0) < 0) 
        errorFlag = @"sysctl mgmtInfoBase failure";
    // Alloc memory based on above call
    else if ((msgBuffer = malloc(length)) == NULL)
        errorFlag = @"buffer allocation failure";
    // Get system information, store in buffer
    else if (sysctl(mgmtInfoBase, 6, msgBuffer, &length, NULL, 0) < 0)
    {
        free(msgBuffer);
        errorFlag = @"sysctl msgBuffer failure";
    }
    else
    {
        // Map msgbuffer to interface message structure
        struct if_msghdr *interfaceMsgStruct = (struct if_msghdr *) msgBuffer;

        // Map to link-level socket structure
        struct sockaddr_dl *socketStruct = (struct sockaddr_dl *) (interfaceMsgStruct + 1);

        // Copy link layer address data in socket structure to an array
        unsigned char macAddress[6];
        memcpy(&macAddress, socketStruct->sdl_data + socketStruct->sdl_nlen, 6);

        // Read from char array into a string object, into traditional Mac address format
        NSString *macAddressString = [NSString stringWithFormat:@"%02X:%02X:%02X:%02X:%02X:%02X",
                                      macAddress[0], macAddress[1], macAddress[2], macAddress[3], macAddress[4], macAddress[5]];
        NSLog(@"Mac Address: %@", macAddressString);

        // Release the buffer memory
        free(msgBuffer);

        return macAddressString;
    }

    // Error...
    NSLog(@"Error: %@", errorFlag);

    return nil;
}

我不明白为什么我可以为自己的答案添加注释,而不能为其他人的答案添加注释?:/
心教堂

1
因为您的声誉不够高。
荣誉

谢谢,看来有些人提出了自己的解决方案变体:ios.biomsoft.com/2011/11/07/determine-mac-address
josh-fuggle 2013年

1

在运行iOS 7.0或更高版本的设备上不再可能,因此无法在Swift中获取MAC地址。

正如苹果所说:

在iOS 7及更高版本中,如果您要求提供iOS设备的MAC地址,系统将返回值02:00:00:00:00:00。如果需要标识设备,请改用UIDevice的identifierForVendor属性。(出于广告目的需要标识符的应用应考虑改用ASIdentifierManager的advertiseIdentifier属性。)


0
#import <sys/socket.h>
#import <net/if_dl.h>
#import <ifaddrs.h>
#import <sys/xattr.h>
#define IFT_ETHER 0x6

...

- (NSString*)macAddress
{
    NSString* result = nil;

    char* macAddressString = (char*)malloc(18);
    if (macAddressString != NULL)
    {
        strcpy(macAddressString, "");

        struct ifaddrs* addrs = NULL;
        struct ifaddrs* cursor;

        if (getifaddrs(&addrs) == 0)
        {
            cursor = addrs;

            while (cursor != NULL)
            {
                if ((cursor->ifa_addr->sa_family == AF_LINK) && (((const struct sockaddr_dl*)cursor->ifa_addr)->sdl_type == IFT_ETHER) && strcmp("en0", cursor->ifa_name) == 0)
                {
                    const struct sockaddr_dl* dlAddr = (const struct sockaddr_dl*) cursor->ifa_addr;
                    const unsigned char* base = (const unsigned char*)&dlAddr->sdl_data[dlAddr->sdl_nlen];

                    for (NSInteger index = 0; index < dlAddr->sdl_alen; index++)
                    {
                        char partialAddr[3];

                        sprintf(partialAddr, "%02X", base[index]);
                        strcat(macAddressString, partialAddr);
                    }
                }

                cursor = cursor->ifa_next;
            }

        }

        result = [[[NSString alloc] initWithUTF8String:macAddressString] autorelease];

        free(macAddressString);
    }

    return result;
}

0

要基于iOS 6中设备的唯一标识符创建uniqueString,请执行以下操作:

#import <AdSupport/ASIdentifierManager.h>

NSString *uniqueString = [[[ASIdentifierManager sharedManager] advertisingIdentifier] UUIDString];
NSLog(@"uniqueString: %@", uniqueString);

1
这与所提问题有什么关系?
Valeriy Van

1
在大多数情况下,需要Mac地址才能唯一标识设备。相反,您可以使用此解决方案。
Denis Kutlubaev

0

其中许多问题仅针对Mac地址。如果您还需要我刚刚写的IP地址,则可能需要做一些工作,但在我的机器上似乎可以正常工作...

- (NSString *)getLocalIPAddress
{
    NSArray *ipAddresses = [[NSHost currentHost] addresses];
    NSArray *sortedIPAddresses = [ipAddresses sortedArrayUsingSelector:@selector(localizedCaseInsensitiveCompare:)];

    NSNumberFormatter *numberFormatter = [[NSNumberFormatter alloc] init];
    numberFormatter.allowsFloats = NO;

    for (NSString *potentialIPAddress in sortedIPAddresses)
    {
        if ([potentialIPAddress isEqualToString:@"127.0.0.1"]) {
            continue;
        }

        NSArray *ipParts = [potentialIPAddress componentsSeparatedByString:@"."];

        BOOL isMatch = YES;

        for (NSString *ipPart in ipParts) {
            if (![numberFormatter numberFromString:ipPart]) {
                isMatch = NO;
                break;
            }
        }
        if (isMatch) {
            return potentialIPAddress;
        }
    }

    // No IP found
    return @"?.?.?.?";
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.