獲取iOS設(shè)備唯一標(biāo)識(shí)

在開(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];

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容