Как я могу программно получить MAC-адрес iphone?

144

Как программно получить MAC-адрес и IP-адрес iPhone?

Кер
источник
Я хочу получить MAC-адрес и IP-адрес программно в приложении для iPhone.
Wifi MAC-адрес? или BlueTooth? Есть много Mac-адресов.
mskw
11
Начиная с iOS 7, система всегда возвращает значение, 02:00:00:00:00:00когда вы запрашиваете MAC-адрес на любом устройстве. Проверьте мой ответ ниже.
Хиджази,
Для СВН и вверх (до сегодняшнего дня) см stackoverflow.com/questions/11836225/...
rptwsthi

Ответы:

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

Названия адаптеров различаются в зависимости от симулятора / устройства, а также от Wi-Fi или сотового устройства.

PyjamaSam
источник
Мне просто интересно, если это отличается от общего подхода OS X (я спрашиваю, потому что я Mac N00B)
Тамас Чинеге
2
@abc: Я рекомендую опубликовать еще один вопрос по этому поводу. Лучше спрятать его в комментариях к другому вопросу.
PyjamaSam
1
а как насчет вызова ether_ntoa? это частный API не так ли?
Стиги
1
@NicTesla Я не могу комментировать наверняка, но, поскольку он просто выполняет базовые сетевые вызовы, я не вижу проблемы. Хотя, как говорится, после удаления UDID из iOS5, получения mac-адреса и использования его для разработки какого-то альтернативного уникального идентификатора, Apple может в будущем стать объектом пристального внимания.
PyjamaSam
1
Что касается ether_ntoaпредупреждения, проверьте: stackoverflow.com/a/13412265/88597
Охо
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;
}
ArtFeel
источник
изменить ссылку на мобильные разработчики вместо iphonedevelopertips
SEG
Apple, очень вероятно, сломает это в iOS7 ... и мы не сможем комментировать это публично, пока она не поступит в продажу, кроме как на официальных форумах Apple Dev.
Гейб
3
Это не сработало на моем iPhone5 с iOS7. Это дает мне: 02: 00: 00: 00: 00: 00, что не правильно.
Sjoerd представляет
3
@Brenden Вы больше не можете получить сетевой MAC-адрес, используя PublicAPI на iOS7. Итак, если вам нужен уникальный идентификатор, вы должны использоватьIdentifierManager
ArtFeel
2
@SjoerdPerfors В iOS 7 вы действительно всегда получите один и тот же адрес, 02:00:00:00:00:00что и меры предосторожности для конфиденциальности Apple.
Василий Бурк
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 должен быть автоматически освобожден, если указанный выше код вызова находится в другом методе.
4
чтобы быть понятным, сам код не будет просачиваться, комментатор говорил о вызывающем фрагменте в конце, где я выделяю две строки, не освобождая их.
капитан корабля
1
Я поместил это в хороший метод target
wsidell
23

Начиная с iOS 7, система всегда возвращает значение, 02:00:00:00:00:00когда вы запрашиваете MAC-адрес на любом устройстве.

В iOS 7 и более поздних версиях, если вы запрашиваете MAC-адрес устройства iOS, система возвращает значение 02: 00: 00: 00: 00: 00. Если вам нужно идентифицировать устройство, используйте вместо него свойство identifierForVendor UIDevice. (Приложения, которым нужен идентификатор для собственных рекламных целей, должны рассмотреть возможность использования свойства advertisingIdentifier ASIdentifierManager.) "

Ссылка: Releasenotes

Hejazi
источник
Что действительно приятно в этом устаревании в API, так это то, что он все еще работает на симуляторе правильно, и только на аппаратном уровне в данный момент.
Джон Най
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!");
}
Kenial
источник
Очень хороший и полезный класс! Хорошая работа! Однако вы пропустили [super dealloc] в обоих классах и забыли включить библиотеку, которая приводит к предупреждениям в xCode. Исправленную версию можно найти здесь: raptor.hk/download/NICInfo_raptor_patched.zip
Raptor
получаю этот результат Mac Адрес: 02: 00: 00: 00: 00: 00
Сталин Пуспарай
1
Да. Apple заблокировала это с 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;
}
Grantland Chew
источник
1
Я использую ваше решение, 99% случаев работают хорошо, как и ожидалось, но 1% пользователей не могут получить MAC-адрес, возвращают NULL, сообщают пользователи в AppStore. Все еще не получить воспроизводимые шаги, у вас есть это? Спасибо.
Цзяньхуа
Я еще не заметил никаких проблем, но я буду следить!
Grantland Chew
В продолжение комментария @ jianhua: мы только что получили отчеты от пользователя, который в принципе не может использовать приложение, потому что этот код возвращает нуль для его устройства (мы используем идентификатор для проверки подлинности при каждом вызове сервера).
Самверметт
Неудачный случай - на iPhone 4S с версией IOS 5.0.1, конечно, это только в некоторых особых случаях.
Цзяньхуа
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);
Нагендра Трипати
источник
будет ли этот идентификатор согласован для каждого устройства в нескольких установках одного и того же приложения?
Самсам
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
Потому что ваша репутация недостаточно высока.
Хонус
Спасибо, похоже, несколько человек придумали свои варианты этого решения: ios.biomsoft.com/2011/11/07/determine-mac-address
josh-fuggle
1

Это больше невозможно на устройствах под управлением iOS 7.0 или новее, поэтому невозможно получить MAC-адрес в Swift.

Как заявило Apple:

В iOS 7 и более поздних версиях, если вы запрашиваете MAC-адрес устройства iOS, система возвращает значение 02: 00: 00: 00: 00: 00. Если вам нужно идентифицировать устройство, используйте вместо него свойство identifierForVendor UIDevice. (Приложения, которым нужен идентификатор для собственных рекламных целей, должны рассмотреть возможность использования свойства advertisingIdentifier ASIdentifierManager.)

pedrouan
источник
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

Чтобы создать uniqueString на основе уникального идентификатора устройства в iOS 6:

#import <AdSupport/ASIdentifierManager.h>

NSString *uniqueString = [[[ASIdentifierManager sharedManager] advertisingIdentifier] UUIDString];
NSLog(@"uniqueString: %@", uniqueString);
Денис Кутлубаев
источник
1
как это связано с заданным вопросом?
Валерий Ван
1
Mac-адрес необходим для уникальной идентификации устройства в большинстве случаев. Вместо этого вы можете использовать это решение.
Денис Кутлубаев
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 @"?.?.?.?";
}
Оливер Пирмейн
источник