Як я можу програмно отримати MAC-адресу iphone


144

Як програмно отримати MAC-адресу та IP-адресу iPhone?


Я хочу отримати MAC-адресу та IP-адресу програмно в додатку iPhone.

MAC-адреса Wifi? чи 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 або стільника на пристрої.


Мені просто цікаво, чи це чим відрізняється від загального підходу до OS X (я прошу, бо я мак n00b)
Тамас Цінеж

2
@abc: Я рекомендую вам написати ще одне питання з цього приводу. Будь краще, ніж поховати його в коментарях до іншого питання.
PyjamaSam

1
а як щодо заклику до ether_ntoa? це приватний API, чи не так?
стигі

1
@NicTesla Я не можу коментувати напевно, але оскільки я просто виконую основні мережеві дзвінки, я не бачу проблеми. Хоча це буде сказано про видалення UDID з iOS5, отримання mac-адреси та використання її для розробки якогось альтернативного унікального ідентифікатора може піддатися ретельному контролю з боку Apple в якийсь момент у майбутньому.
PyjamaSam

1
Що стосується ether_ntoaпопередження, перевірте: stackoverflow.com/a/13412265/88597
ohho

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

Apple, ДУЖЕ, ймовірно, зламає це в iOS7 ... і ми не зможемо коментувати це публічно, поки воно не надходить, окрім офіційних форумів Apple Dev.
Гейб

3
На моєму iPhone5 з iOS7 це не працювало. Це дає мені: 02: 00: 00: 00: 00: 00: 00, що невірно.
Шьорд Перфорс

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
Я вклав це в приємний метод-c і зробив це таким чином, щоб він не потребував параметрів: gist.github.com/wsidell/5069159
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.) "

Довідка: релізні примітки


Що насправді приємно в цій депресії в 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!");
}

Дуже приємний та корисний клас! Гарна робота! Однак ви пропустили [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;
}

1
Я використовую ваше рішення, 99% випадків працюють добре, як очікувалося, але 1% ставка не може отримати mac-адресу, повернути NULL, повідомили користувачі в AppStore. Ви все ще не отримали відтворюваних сходинок, чи це ви отримали? Дякую.
Цзяньхуа

Я ще не помітив жодних проблем, але я буду тримати увагу!
Grantland Chew

Слідкуючи за коментарем @ jianhua: ми щойно отримували звіти від користувача, який в основному не може використовувати програму, оскільки цей код повертає нуль для свого пристрою (ми використовуємо ідентифікатор для автентифікації кожного дзвінка на сервері).
samvermette

Невдалий випадок є на iPhone 4S з IOS версії 5.0.1, звичайно, це є лише в деяких особливих випадках.
jianhua

5

Тепер пристрої iOS 7 - завжди повертають MAC-адресу 02: 00: 00: 00: 00: 00: 00.

Тому краще скористайтеся [ідентифікатором UIDeviceForVendor].

тому краще зателефонувати за цим методом, щоб отримати специфічний для програми унікальний ключ

Категорія більше підходить

імпортувати "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

Дякую, схоже, що мало хто придумав свої варіанти цього рішення: 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.)


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

Щоб створити унікальнуString на основі унікального ідентифікатора пристрою в 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 @"?.?.?.?";
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.