在開(kāi)發(fā)過(guò)程中,我們經(jīng)常會(huì)被要求獲取每個(gè)設(shè)備的唯一標(biāo)示,以便后臺(tái)做相應(yīng)的處理。我們來(lái)看看有哪些方法來(lái)獲取設(shè)備的唯一標(biāo)示,然后再分析下這些方法的利弊。
具體可以分為如下幾種:1、UUID2、IDFA3、IDFV4、MAC5、UUID
下面我們來(lái)具體分析下每種獲取方法的利弊
1、UDID
什么是UDID
UDID 「Unique Device Identifier Description」是由子母和數(shù)字組成的40個(gè)字符串的序號(hào),用來(lái)區(qū)別每一個(gè)唯一的iOS設(shè)備,包括 iPhones, iPads, 以及 iPod touches,這些編碼看起來(lái)是隨機(jī)的,實(shí)際上是跟硬件設(shè)備特點(diǎn)相聯(lián)系的。
UDID是用來(lái)干什么的?
UDID可以關(guān)聯(lián)其它各種數(shù)據(jù)到相關(guān)設(shè)備上。例如,連接到開(kāi)發(fā)者賬號(hào),可以允許在發(fā)布前讓設(shè)備安裝或測(cè)試應(yīng)用;也可以讓開(kāi)發(fā)者獲得iOS測(cè)試版進(jìn)行體驗(yàn)。蘋果用UDID連接到蘋果的ID,這些設(shè)備可以自動(dòng)下載和安裝從App Store購(gòu)買的應(yīng)用、保存從iTunes購(gòu)買的音樂(lè)、幫助蘋果發(fā)送推送通知、即時(shí)消息。 在iOS 應(yīng)用早期,UDID被第三方應(yīng)用開(kāi)發(fā)者和網(wǎng)絡(luò)廣告商用來(lái)收集用戶數(shù)據(jù),可以用來(lái)關(guān)聯(lián)地址、記錄應(yīng)用使用習(xí)慣……以便推送精準(zhǔn)廣告。
為什么蘋果反對(duì)開(kāi)發(fā)人員使用UDID?
iOS 2.0版本以后UIDevice提供一個(gè)獲取設(shè)備唯一標(biāo)識(shí)符的方法uniqueIdentifier,通過(guò)該方法我們可以獲取設(shè)備的序列號(hào),這個(gè)也是目前為止唯一可以確認(rèn)唯一的標(biāo)示符。 許多開(kāi)發(fā)者把UDID跟用戶的真實(shí)姓名、密碼、住址、其它數(shù)據(jù)關(guān)聯(lián)起來(lái);網(wǎng)絡(luò)窺探者會(huì)從多個(gè)應(yīng)用收集這些數(shù)據(jù),然后順藤摸瓜得到這個(gè)人的許多隱私數(shù)據(jù)。同時(shí)大部分應(yīng)用確實(shí)在頻繁傳輸U(kuò)DID和私人信息。 為了避免集體訴訟,蘋果最終決定在iOS 5 的時(shí)候,將這一慣例廢除,開(kāi)發(fā)者被引導(dǎo)生成一個(gè)唯一的標(biāo)識(shí)符,只能檢測(cè)應(yīng)用程序,其他的信息不提供。現(xiàn)在應(yīng)用試圖獲取UDID已被禁止且不允許上架。
所以這個(gè)方法作廢。
2、IDFA
全名:advertisingIdentifier
獲取代碼:#import<AdSupport/AdSupport.h>
NSString *adId = [[[ASIdentifierManager sharedManager] advertisingIdentifier] UUIDString];
來(lái)源:iOS6.0及以后
說(shuō)明:直譯就是廣告id, 在同一個(gè)設(shè)備上的所有App都會(huì)取到相同的值,是蘋果專門給各廣告提供商用來(lái)追蹤用戶而設(shè)的,用戶可以在 設(shè)置|隱私|廣告追蹤 里重置此id的值,或限制此id的使用,故此id有可能會(huì)取不到值,但好在Apple默認(rèn)是允許追蹤的,而且一般用戶都不知道有這么個(gè)設(shè)置,所以基本上用來(lái)監(jiān)測(cè)推廣效果,是戳戳有余了。
注意:由于idfa會(huì)出現(xiàn)取不到的情況,故絕不可以作為業(yè)務(wù)分析的主id,來(lái)識(shí)別用戶。
3、IDFV
全名:identifierForVendor
獲取代碼:
NSString *idfv = [[[UIDevice currentDevice] identifierForVendor] UUIDString];
來(lái)源:iOS6.0及以后
說(shuō)明:顧名思義,是給Vendor標(biāo)識(shí)用戶用的,每個(gè)設(shè)備在所屬同一個(gè)Vender的應(yīng)用里,都有相同的值。其中的Vender是指應(yīng)用提供商,但準(zhǔn)確點(diǎn)說(shuō),是通過(guò)BundleID的反轉(zhuǎn)的前兩部分進(jìn)行匹配,如果相同就是同一個(gè)Vender,例如對(duì)于com.taobao.app1, com.taobao.app2 這兩個(gè)BundleID來(lái)說(shuō),就屬于同一個(gè)Vender,共享同一個(gè)idfv的值。和idfa不同的是,idfv的值是一定能取到的,所以非常適合于作為內(nèi)部用戶行為分析的主id,來(lái)標(biāo)識(shí)用戶,替代OpenUDID。
注意:如果用戶將屬于此Vender的所有App卸載,則idfv的值會(huì)被重置,即再重裝此Vender的App,idfv的值和之前不同。
4、MAC地址
使用WiFi的mac地址來(lái)取代已經(jīng)廢棄了的uniqueIdentifier方法。具體可見(jiàn): http://stackoverflow.com/questions/677530/how-can-i-programmatically-get-the-mac-address-of-an-iphone
然而在iOS 7中蘋果再一次無(wú)情的封殺mac地址,使用之前的方法獲取到的mac地址全部都變成了02:00:00:00:00:00。
5、UUID
我們可以獲取到UUID,然后把UUID保存到KeyChain里面。
這樣以后即使APP刪了再裝回來(lái),也可以從KeyChain中讀取回來(lái)。使用group還可以可以保證同一個(gè)開(kāi)發(fā)商的所有程序針對(duì)同一臺(tái)設(shè)備能夠獲取到相同的不變的UDID。
但是刷機(jī)或重裝系統(tǒng)后uuid還是會(huì)改變。
使用Keychain,將UUID當(dāng)做密碼信息來(lái)存儲(chǔ).
大致流程:
通過(guò)AdSupport獲取UUID(原因AdSupport可以跨應(yīng)用)
1. (NSString *)appleIFA {
NSString *ifa = nil;
Class ASIdentifierManagerClass = NSClassFromString(@"ASIdentifierManager");
if (ASIdentifierManagerClass) { // a dynamic way of checking if AdSupport.framework is available
SEL sharedManagerSelector = NSSelectorFromString(@"sharedManager");
id sharedManager = ((id (*)(id, SEL))[ASIdentifierManagerClass methodForSelector:sharedManagerSelector])(ASIdentifierManagerClass, sharedManagerSelector);
SEL advertisingIdentifierSelector = NSSelectorFromString(@"advertisingIdentifier");
NSUUID *advertisingIdentifier = ((NSUUID* (*)(id, SEL))[sharedManager methodForSelector:advertisingIdentifierSelector])(sharedManager, advertisingIdentifierSelector);
ifa = [advertisingIdentifier UUIDString];
}
return ifa;
}
如果不支持AdSupport,那就使用IFV/IDFV (Identifier for Vendor)
+ (NSString *)appleIFV {
if(NSClassFromString(@"UIDevice") && [UIDevice instancesRespondToSelector:@selector(identifierForVendor)]) {
// only available in iOS >= 6.0
return [[UIDevice currentDevice].identifierForVendor UUIDString];
}
return nil;
}
如果以上的都不支持,使用CFUUIDRef手動(dòng)創(chuàng)建UUID
+ (NSString *)randomUUID {
if(NSClassFromString(@"NSUUID")) { // only available in iOS >= 6.0
return [[NSUUID UUID] UUIDString];
}
CFUUIDRef uuidRef = CFUUIDCreate(kCFAllocatorDefault);
CFStringRef cfuuid = CFUUIDCreateString(kCFAllocatorDefault, uuidRef);
CFRelease(uuidRef);
NSString *uuid = [((__bridge NSString *) cfuuid) copy];
CFRelease(cfuuid);
return uuid;
}
最后,添加到Keychain
+ (void)setValue:(NSString *)value forKey:(NSString *)key inService:(NSString *)service {
NSMutableDictionary *keychainItem = [[NSMutableDictionary alloc] init];
keychainItem[(__bridge id)kSecClass] = (__bridge id)kSecClassGenericPassword;
keychainItem[(__bridge id)kSecAttrAccessible] = (__bridge id)kSecAttrAccessibleAlways;
keychainItem[(__bridge id)kSecAttrAccount] = key;
keychainItem[(__bridge id)kSecAttrService] = service;
keychainItem[(__bridge id)kSecValueData] = [value dataUsingEncoding:NSUTF8StringEncoding];
SecItemAdd((__bridge CFDictionaryRef)keychainItem, NULL);
}
如果更新了provisioning profile的話, Keychain data會(huì)丟失.所以我們應(yīng)該將UUID在NSUserDefault備份.
[[NSUserDefaults standardUserDefaults] setObject:@”123456-1234-1234-12345678” forKey:@"deviceUID"];
[[NSUserDefaults standardUserDefaults] synchronize];