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

146

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


我想在iPhone应用程序中以编程方式获取MAC地址和IP地址。 - abc
Wifi的MAC地址?还是蓝牙的?有很多个MAC地址。 - mskw
12
从iOS 7开始,在任何设备上请求MAC地址时,系统始终返回值为“02:00:00:00:00:00”。请确认我的答案是否正确。 - Hejazi
针对 iOS 系统及其更新版本(截至今日),请参考 https://dev59.com/jGgt5IYBdhLWcg3wwwK-。 - rptwsthi
12个回答

116

注意:自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或蜂窝网络,适配器名称会有所不同。


2
@abc:我建议你另外发一个问题来询问。这样比在其他问题的评论中埋没回答会更好。 - PyjamaSam
1
但是对于 ether_ntoa 的调用怎么办?这是私有 API,不是吗? - stigi
1
@NicTesla 我不能确定,但由于它只是进行基本的网络调用,我看不出有什么问题。尽管如此,随着 iOS5 中 UDID 的移除,获取 MAC 地址并使用它来开发某种替代唯一标识符可能会在未来受到苹果公司的审查。 - PyjamaSam
1
关于 ether_ntoa 警告,请查看:https://dev59.com/1WPVa4cB1Zd3GeqP4lEw#13412265 - ohho
请注意,该 C 代码的版权声明为“保留所有权利”。 - jeff7091
显示剩余7条评论

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;
}

3
这在我使用iOS7的iPhone5上没有生效。它给了我"02:00:00:00:00:00",这不正确。 - Sjoerd Perfors
请看我的上面的评论。 - ArtFeel
@ArtFeel ASIdentifierManager不提供网络MAC地址。 - Brenden
3
在 iOS7 上,无法使用 PublicAPI 获取网络MAC地址。因此,如果需要唯一标识符,应该使用“IdentifierManager”。 - ArtFeel
2
在iOS 7下,由于苹果的隐私保护措施,您将始终获得相同的地址“02:00:00:00:00:00”。@SjoerdPerfors - Basil Bourque
显示剩余3条评论

31

我希望能够返回地址,无论Wi-Fi是否启用,因此所选择的解决方案对我来说不起作用。经过一些调整后,我在某个论坛上找到了另一个调用。最终我得到了以下结果(请原谅我的生疏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
这个对我不起作用... 说ifName未被声明,以及macaddress变量。 - bugfixr
以上的调用代码会造成内存泄漏。在从上述调用方法返回之前,应该释放macAddressString。此外,如果上述调用代码在另一个方法中,则macAddress应该是自动释放的。 - user988505
@Veera.s.Vasan,你是正确的。这个片段并不意味着是完整的代码,只是为了向你展示如何使用上面的函数。 - shipmaster
4
要明确的是,代码本身不会泄漏,评论者所说的是在结尾处调用片段中,我分配了两个字符串而没有释放它们。 - shipmaster
1
我将这段代码封装成一个漂亮的Objective-C方法,并且使其不需要任何参数:https://gist.github.com/wsidell/5069159 - wsidell

23

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

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

参考资料:发布说明


这个API中的弃用之处非常好的一点是它仍然可以在模拟器上正确地工作,目前只在硬件上存在问题。 - John Nye

12

有很多解决方案,但我找不到一个完整的。 所以我为此做了自己的解决方案:

nicinfo

如何使用:

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中出现警告。修补版本可以在这里找到:http://raptor.hk/download/NICInfo_raptor_patched.zip - Raptor
获得以下结果的Mac地址:02:00:00:00:00:00 - Stalin Pusparaj
1
是的,自从iOS 7以后,苹果就阻止了这个功能 :( ... - 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中由用户报告。现在仍然找不到可重现的步骤,你们有吗?谢谢。 - jianhua
我还没有注意到任何问题,但我会继续留意! - Grantland Chew
针对@jianhua的评论进行跟进:我们刚刚收到了一位用户的报告,他基本上无法使用该应用程序,因为此代码在他的设备上返回null(我们使用标识符来验证每个服务器调用)。 - samvermette
失败的情况出现在IOS版本为5.0.1的iPhone 4S上,当然这只是在一些特殊场合下发生。 - jianhua

5

现在,iOS 7设备总是返回MAC地址02:00:00:00:00:00。

因此,最��使用[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

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;
}

我不明白为什么我可以为自己的答案添加评论,但不能为别人的答案添加评论? :/ - Cœur
1
因为您的声誉还不够高。 - honus
谢谢,看起来有一些人已经提出了自己的解决方案:http://ios.biomsoft.com/2011/11/07/determine-mac-address/。 - josh-fuggle

1
在运行iOS 7.0或更高版本的设备上,无法获取Swift中的MAC地址,因此不再可能。正如苹果公司所述:在iOS 7及更高版本中,如果您请求iOS设备的MAC地址,则系统会返回值02:00:00:00:00:00。如果您需要识别设备,请改用UIDevice的identifierForVendor属性。(需要自己广告目的的应用程序应考虑使用ASIdentifierManager的advertisingIdentifier属性。)

0

在iOS 6中,基于设备唯一标识符创建一个uniqueString的方法如下:

#import <AdSupport/ASIdentifierManager.h>

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

1
这与所提出的问题有何关联? - Valeriy Van
1
大多数情况下,需要使用MAC地址来唯一标识设备。但是,您可以使用这个解决方案。 - Denis Kutlubaev

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接