プログラムでiphoneのMACアドレスを取得する方法


144

プログラムでiPhoneのMACアドレスとIPアドレスを取得する方法は?


iPhoneアプリケーションでプログラムでMACアドレスとIPアドレスを取得したい。

Wifi MACアドレス?またはBlueTooth?多くのMacアドレスがあります。
mskw 2013年

11
iOS 7以降、02:00:00:00:00:00デバイスでMACアドレスを要求すると、システムは常に値を返します。以下の私の答えを確認してください。
Hejazi 2013年

回答:


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 2009年

2
@abc:別の質問を投稿することをお勧めします。別の質問のコメントに埋め込んでください。
PyjamaSam 2009年

1
しかし、ether_ntoaの呼び出しはどうですか?これはプライベートAPIですね。
stigi 2009年

1
@NicTesla確かにコメントすることはできませんが、基本的なネットワーク呼び出しを行っているだけなので、問題はありません。iOS5からUDIDが削除されたと言われていますが、Macアドレスを取得し、それを使用してある種の代替の一意の識別子を開発することは、将来のある時点でAppleによって調査される可能性があります。
PyjamaSam 2011

1
ether_ntoa警告については、チェックアウトしてください:stackoverflow.com/a/13412265/88597
ohho

45

更新:これはiOS 7では機能しません。ASIdentifierManagerを使用する必要があります。


MobileDeveloperTips Webサイトのよりクリーンなソリューション

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

リンクをiphonedevelopertipsではなくmobiledevelopertipsに変更します
SEG

AppleはiOS7でこれを破る可能性が非常に高いです...そして、公式のApple Dev Forumsを除いて出荷されるまで、これについて公にコメントすることはできません。
Gabe 2013

3
これはiOS7を搭載したiPhone5では機能しませんでした。それは私に与える:02:00:00:00:00:00これは正しくありません。
Sjoerd Perfors 2013

3
@Brenden iOS7でPublicAPIを使用してネットワークMACアドレスを取得できなくなりました。したがって、一意のIDが必要な場合は、使用する必要がありますIdentifierManager
ArtFeel

2
@SjoerdPerfors iOS 7では02:00:00:00:00:00、Appleによるプライバシーの予防措置として、常に同じアドレスを取得します。
バジルブルク2014年

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

これは私にはうまくいきません... ifNameが宣言されていないだけでなく、macaddress変数も宣言されていません。
bugfixr

上記の呼び出しコードはリークします。free(macAddressString); 上記の呼び出しメソッドから戻る前。また、上記の呼び出しコードが別のメソッドにある場合は、macAddressを自動解放する必要があります。

4
明確にするために、コード自体はリークしません。コメンターは最後に呼び出しスニペットについて話していました。ここでは、2つの文字列を解放せずに割り当てています。
船長、2011

1
私はこれをnice-objective-cメソッドに入れてパラメーターを必要としないように作成しました:gist.github.com/wsidell/5069159
wsidell

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

これにはさまざまな解決策がありますが、すべてを見つけることはできませんでした。だから私は自分用のソリューションを作りました:

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
スターリンプスパラジ2017

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のユーザーから報告されます。まだ再現可能なステップが得られないのですが、わかりましたか?ありがとうございました。
建華

私はまだ問題に気づいていませんが、今後も注目していきます!
Grantland Chew、

@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

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
あなたの評判が十分に高くないからです。
2012年

おかげで、このソリューションの独自のバリアントをいくつかの人々が思いついたようです: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
#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 2013

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 @"?.?.?.?";
}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.