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

141

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

  • 0
    я хочу получить MAC-адрес и IP-адрес программно в приложении для iPhone.
  • 1
    используя цель-C?
Показать ещё 3 комментария
Теги:
iphone
mac-address

12 ответов

117
Лучший ответ

ПРИМЕЧАНИЕ. Начиная с 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 или ячейки на устройстве.

  • 0
    Мне просто интересно, если это отличается от общего подхода OS X (я спрашиваю, потому что я Mac N00B)
  • 0
    @DrJokepu Я думаю, что это основной OSX материал. На самом деле его основной материал BSD. Тот же подход, скорее всего, будет работать и на Linux (даже, возможно, с небольшими изменениями в Windows)
Показать ещё 12 комментариев
43

Обновление: это не будет работать на 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;
}
  • 0
    Ссылка недействительна
  • 0
    у меня все отлично работает
Показать ещё 8 комментариев
32

Я хотел что-то вернуть адрес независимо от того, включен ли 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
  • 0
    Этот не работает для меня ... говорит, что ifName не объявлен так же, как переменная macaddress.
Показать ещё 4 комментария
24

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

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

Ссылка: releasenotes

  • 0
    Что действительно приятно в этом устаревшем 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!");
}
  • 0
    Очень хороший и полезный класс! Хорошая работа! Однако вы пропустили [super dealloc] в обоих классах и забыли включить библиотеку, которая приводит к предупреждениям в xCode. Исправленную версию можно найти здесь: raptor.hk/download/NICInfo_raptor_patched.zip
  • 1
    Спасибо, я подам это прямо сейчас :)
Показать ещё 2 комментария
5

Теперь устройства iOS 7 - всегда возвращают MAC-адрес 02: 00: 00: 00: 00: 00.

Поэтому лучше использовать [UIDevice identifierForVendor].

поэтому лучше вызвать этот метод, чтобы получить уникальный уникальный для приложения ключ

Категория будет более подходящей

import "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);
  • 0
    будет ли этот идентификатор согласован для каждого устройства в нескольких установках одного и того же приложения?
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;
}
  • 0
    +1 за сам проект Гекица
  • 1
    Я использую ваше решение, 99% случаев работают хорошо, как и ожидалось, но 1% пользователей не могут получить MAC-адрес, возвращают NULL, сообщают пользователи в AppStore. Все еще не получить воспроизводимые шаги, у вас есть это? Спасибо.
Показать ещё 3 комментария
2

@Grantland Это "довольно чистое решение" похоже на мое собственное улучшение по сравнению с решением iPhoneDeveloperTips.

Вы можете увидеть мой шаг здесь: https://gist.github.com/1409855/

  • 0
    и я не понимаю, почему я могу добавлять комментарии для своих собственных ответов, но не для ответов других? : /
  • 1
    Потому что ваша репутация недостаточно высока.
Показать ещё 1 комментарий
0

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

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

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

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 @"?.?.?.?";
}
0

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

#import <AdSupport/ASIdentifierManager.h>

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

Ещё вопросы

Сообщество Overcoder
Наверх
Меню