iOS设备获取唯一标志符

背景

  近期公司app要加入类似统计的功能,把设备的唯一标志符传递给服务器,用以统计未登录用户的操作行为。所以做了一些调研,把iOS中的各种可能碰到的标志符的定义及使用情况罗列出来,最后附上获取唯一标志符的办法。

几大标志符

1、UDID(Unique Device Identifier)

  UDID的全称是Unique Device Identifier,它就是苹果IOS设备的唯一识别码,它由40个字符的字母和数字组成。移动网络可利用UDID来识别移动设备,但是,从IOS5.0(2011年8月份)开始,苹果宣布将不再支持用uniqueIdentifier方法获取设备的UDID,iOS5以下是可以用的。在2013年3月21日苹果已经通知开发者:从2013年5月1日起,访问UIDIDs的程序将不再被审核通过,替代的方案是开发者应该使用“在iOS 6中介绍的Vendor或Advertising标示符”。
  UDID可以在iTunes中查看,多次点击ECID后,该位置会显示为UDID,即为设备的唯一标志符,且永久不变(当然越狱设备通过某种手段还是可以修改的),UDID可以在ipa包的测试配置文件中添加,以达到多人测试的目的。


iOS设备获取唯一标志符_第1张图片
UDID获取方式
2、UUID(Universally Unique Identifier)

  UUID是Universally Unique Identifier的缩写,中文意思是通用唯一识别码。它是让分布式系统中的所有元素,都能有唯一的辨识资讯,而不需要透过中央控制端来做辨识资讯的指定。这样,每个人都可以建立不与其它人冲突的 UUID。在此情况下,就不需考虑数据库建立时的名称重复问题。苹果公司建议使用UUID为应用生成唯一标识字符串。每次生成都会改变。
  生成方法:

- (NSString *) uuid { 
    CFUUIDRef puuid = CFUUIDCreate( nil ); 
    CFStringRef uuidString = CFUUIDCreateString( nil, puuid );
    NSString * result = (NSString *)CFStringCreateCopy( NULL, uuidString); 
    CFRelease(puuid); 
    CFRelease(uuidString); 
    return [result autorelease]; 
}
3、Vendor标示符 (IDFV-identifierForVendor)

  Vendor标示符,也是在iOS 6中新增的,跟advertisingIdentifier一样,该方法返回的是一个 NSUUID对象,可以获得一个UUID。如果满足条件“相同的一个程序里面-相同的vendor-相同的设备”,那么获取到的这个属性值就不会变。如果是“相同的程序-相同的设备-不同的vendor,或者是相同的程序-不同的设备-无论是否相同的vendor”这样的情况,那么这个值是不会相同的。

NSUUID * currentDeviceUUID  = [UIDevice currentDevice].identifierForVendor;

  以上就是获取Vendor标志符方法,返回一个UUID(UUID是标志符的一个种类,而Vendor标志符是UUID的一个子类)。
这里生成的是NSUUID,这里解释下CFUUID和NSUUID。
  CFUUID从iOS2.0开始,CFUUID就已经出现了。它是CoreFoundatio包的一部分,因此API属于C语言风。
  NSUUID在iOS 6中才出现,这跟CFUUID几乎完全一样,只不过它是Objective-C接口。

4、MAC Address

  MAC(Medium/Media Access Control)地址,用来表示互联网上每一个站点的标识符,采用十六进制数表示,共六个字节(48位)。其中,前三个字节是由IEEE的注册管理机构 RA负责给不同厂家分配的代码(高位24位),也称为“编制上唯一的标识符” (Organizationally Unique Identifier),后三个字节(低位24位)由各厂家自行指派给生产的适配器接口,称为扩展标识符(唯一性)。
  MAC地址在网络上用来区分设备的唯一性,接入网络的设备都有一个MAC地址,他们肯定都是不同的,是唯一的。一部iPhone上可能有多个MAC地址,包括WIFI的、SIM的等,但是iTouch和iPad上就有一个WIFI的,因此只需获取WIFI的MAC地址就好了,也就是en0的地址。
  形象的说,MAC地址就如同我们身份证上的身份证号码,具有全球唯一性。这样就可以非常好的标识设备唯一性,类似与苹果设备的UDID号,通常的用途有:1)用于一些统计与分析目的,利用用户的操作习惯和数据更好的规划产品;2)作为用户ID来唯一识别用户,可以用游客身份使用app又能在服务器端保存相应的信息,省去用户名、密码等注册过程。
  iOS7之前,因为Mac地址是唯一的, 一般app开发者会采取获取MAC地址的方式来识别安装对应app的设备。但iOS7之后,获取到的MAC地址全部都变为02:00:00:00:00:00,也就不存在唯一标志一说了。
  获取MAC地址的方法:

#include   
#include   
#include 

- (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");  
        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)];  
      
//    NSString *outstring = [NSString stringWithFormat:@"%02x%02x%02x%02x%02x%02x", *ptr, *(ptr+1), *(ptr+2), *(ptr+3), *(ptr+4), *(ptr+5)];  
      
    NSLog(@"outString:%@", outstring);  
      
    free(buf);  
      
    return [outstring uppercaseString];  
}  
5、OPEN UDID

  OPEN UDID能保证同一台设备上的不同应用使用同一个OpenUDID,只要用户设备上有一个使用了OpenUDID的应用存在时,其他后续安装的应用如果获取OpenUDID,都将会获得第一个应用生成的那个。但是如果把使用了OpenUDID的应用全部都删除,再重新获取OpenUDID,此时的OpenUDID就跟以前的不一样。

6、广告标示符(IDFA-identifierForIdentifier)

  广告标示符,是iOS 6中另外一个新的方法,提供了一个方法advertisingIdentifier,通过调用该方法会返回一个NSUUID实例,最后可以获得一个UUID,由系统存储着的。不过即使这是由系统存储的,但是有几种情况下,会重新生成广告标示符。如果用户完全重置系统((设置程序 -> 通用 -> 还原 -> 还原位置与隐私) ,这个广告标示符会重新生成。另外如果用户明确的还原广告(设置程序-> 通用 -> 关于本机 -> 广告 -> 还原广告标示符) ,那么广告标示符也会重新生成。关于广告标示符的还原,有一点需要注意:如果程序在后台运行,此时用户“还原广告标示符”,然后再回到程序中,此时获取广 告标示符并不会立即获得还原后的标示符。必须要终止程序,然后再重新启动程序,才能获得还原后的广告标示符。
  但IDFA在上架审核的时候,是比较严格的。友盟的统计功能,就使用了IDFA。


iOS设备获取唯一标志符_第2张图片
友盟统计SDK

  在上架审核的时候,会有警告,且审核会有被拒的可能。


iOS设备获取唯一标志符_第3张图片
上架
7、Device Token

  apple push token保证设备唯一,但必须有网络情况下才能工作,该方法不依赖于设备本身,但依赖于apple push,而苹果push有时候会抽风的。

解决方案

  综合以上方案信息,我们会发现,iOS7及之后的系统,是没有办法获取一个类似UDID这样的能保证设备唯一的标志符的。
  所以我们只能通过KeyChain的方式来保存我们获取的UUID。KeyChain(钥匙串)是使用苹果设备经常使用的,通常要调试的话,都得安装证书之类的,这些证书就是保存在KeyChain中,还有我们平时浏览网页记录的账号密码也都是记录在KeyChain中。iOS中的KeyChain相比OS X比较简单,整个系统只有一个KeyChain,每个程序都可以往KeyChain中记录数据,而且只能读取到自己程序记录在KeyChain中的数据。钥匙串的访问需要Security.framework。钥匙串的API都比较恶心,这里推荐使用SAMKeyChain, https://github.com/soffes/SAMKeychain
  具体实现代码就这些:

+ (NSString *)getUniqueStrByUUID {
    NSString * currentDeviceUUIDStr = [SAMKeychain passwordForService:@" " account:@"GreatChefUUID"];
    if (currentDeviceUUIDStr == nil || [currentDeviceUUIDStr isEqualToString:@""])
    {
        NSUUID * currentDeviceUUID  = [UIDevice currentDevice].identifierForVendor;
        currentDeviceUUIDStr = currentDeviceUUID.UUIDString;
        currentDeviceUUIDStr = [currentDeviceUUIDStr stringByReplacingOccurrencesOfString:@"-" withString:@""];
        currentDeviceUUIDStr = [currentDeviceUUIDStr lowercaseString];
        [SAMKeychain setPassword: currentDeviceUUIDStr forService:@" " account:@"GreatChefUUID"];
    }
    return currentDeviceUUIDStr;
}

  当然这也是有弊端的,比如刷机、恢复出厂设置这类会重置手机信息的,都会清空钥匙串信息,我们保存的UUID也就跟着清除了,不过苹果各种限制,我们能做到这种底部,已经很不错了!!!

你可能感兴趣的:(iOS设备获取唯一标志符)