8種iOS獲取設(shè)備唯一標(biāo)識(shí)的方法丈攒,希望對(duì)大家有用哩罪。
UDID
UDID(Unique Device Identifier),iOS 設(shè)備的唯一識(shí)別碼巡验,是一個(gè)40位十六進(jìn)制序列(越獄的設(shè)備通過(guò)某些工具可以改變?cè)O(shè)備的 UDID)际插,移動(dòng)網(wǎng)絡(luò)可以利用 UDID 來(lái)識(shí)別移動(dòng)設(shè)備。
許多開發(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í)在頻繁傳輸 UDID 和私人信息指攒。 為了避免集體訴訟慷妙,蘋果最終決定在 iOS 5 的時(shí)候,將這一慣例廢除允悦。
現(xiàn)在應(yīng)用試圖獲取 UDID 已被禁止且不允許上架膝擂。
MAC 地址
MAC(Medium / Media Access Control)地址,用來(lái)表示互聯(lián)網(wǎng)上每一個(gè)站點(diǎn)的標(biāo)示符澡屡,是一個(gè)六個(gè)字節(jié)(48位)的十六進(jìn)制序列猿挚。前三個(gè)字節(jié)是由 IEEE 的注冊(cè)管理機(jī)構(gòu) RA 負(fù)責(zé)給不同廠家分配的”編制上唯一的標(biāo)示符(Organizationally Unique Identifier)”,后三個(gè)字節(jié)由各廠家自行指派給生產(chǎn)的適配器接口驶鹉,稱為擴(kuò)展標(biāo)示符绩蜻。
MAC 地址在網(wǎng)絡(luò)上用來(lái)區(qū)分設(shè)備的唯一性,接入網(wǎng)絡(luò)的設(shè)備都有一個(gè)MAC地址室埋,他們肯定都是唯一的办绝。一部 iPhone 上可能有多個(gè) MAC 地址伊约,包括 WIFI 的、SIM 的等孕蝉,但是 iTouch 和 iPad 上就有一個(gè) WIFI 的屡律,因此只需獲取 WIFI 的 MAC 地址就好了。一般會(huì)采取 MD5(MAC 地址 + bundleID)獲取唯一標(biāo)識(shí)降淮。
但是 MAC 地址和 UDID 一樣超埋,存在隱私問(wèn)題, iOS 7 之后佳鳖,所有設(shè)備請(qǐng)求 MAC 地址會(huì)返回一個(gè)固定值霍殴,這個(gè)方法也不攻自破了。
OpenUDID
UDID 被棄用后系吩,廣大開發(fā)者需要尋找一個(gè)可以替代的 UDID来庭,并且不受蘋果控制的方案,由此穿挨,OpenUDID 成為了當(dāng)時(shí)使用最廣泛的開源 UDID 代替方案月弛。OpenUDID 利用一個(gè)非常巧妙的方法在不同程序間存儲(chǔ)標(biāo)示符:在粘貼板中用了一個(gè)特殊的名稱來(lái)存儲(chǔ)標(biāo)示符,通過(guò)這種方法科盛,其他應(yīng)用程序也可以獲取帽衙。
蘋果在 iOS 7 之后對(duì)粘貼板做了限制,導(dǎo)致同一個(gè)設(shè)備上的應(yīng)用間土涝,無(wú)法再共享一個(gè) OpenUDID佛寿。
UUID + 自己存儲(chǔ)
UUID(Universally Unique IDentifier)幌墓,通用唯一標(biāo)示符但壮,是一個(gè)32位的十六進(jìn)制序列,使用小橫線來(lái)連接:8-4-4-4-12常侣,通過(guò) NSUUID(iOS 6 之后)[NSUUID UUID].UUIDString 或者 CFUUID(iOS 2 之后) CFBridgingRelease(CFUUIDCreateString(kCFAllocatorDefault, CFUUIDCreate(kCFAllocatorDefault))) 來(lái)獲取蜡饵,但是每次獲取的值都不一樣,需要自己存儲(chǔ)胳施。
推送 token + bundleID
推送 token 保證設(shè)備唯一溯祸,但是必須有網(wǎng)絡(luò)情況下才能工作,該方法不依賴于設(shè)備本身舞肆,但依賴于 apple push焦辅,而 apple push 有時(shí)候會(huì)抽風(fēng)的。
IDFA
IDFA-identifierForIdentifier(廣告標(biāo)示符)椿胯,在同一個(gè)設(shè)備上的所有 APP 都會(huì)取到相同的值筷登,是蘋果專門給各廣告提供商用來(lái)追蹤用戶而設(shè)定的。雖然 iPhone 默認(rèn)是允許追蹤的哩盲,而且一般用戶都不知道有這么個(gè)設(shè)置前方,但是用戶可以在 設(shè)置 - 隱私 - 廣告追蹤 里重置此 ID 的值狈醉,或者限制此 ID 的使用,所以有可能會(huì)取不到值惠险。
IDFV
IDFV-identifierForVendor(Vendor 標(biāo)示符)绍刮,通過(guò) [UIDevice currentDevice].identifierForVendor.UUIDString 來(lái)獲取帮毁。是通過(guò) bundleID 的反轉(zhuǎn)的前兩部分進(jìn)行匹配,如果相同是同一個(gè) Vendor ,例如對(duì)于 com.mayan.app_1 和 com.mayan.app_2 這兩個(gè) bundleID 來(lái)說(shuō)非剃,就屬于同一個(gè) Vendor ,共享同一個(gè) IDFV嗦嗡,和 IDFA 不同的是胆剧,IDFV 的值一定能取到的,所以非常適合于作為內(nèi)部用戶行為分析的主 ID 來(lái)識(shí)別用戶遥缕。但是用戶刪除了該 APP 卫袒,則 IDFV 值會(huì)被重置,再次安裝此 APP 单匣,IDFV 的值和之前的不同夕凝。
IDFV + keychain
通過(guò)以上幾種儲(chǔ)存唯一標(biāo)識(shí)的方法的分析,總結(jié)一下各有優(yōu)劣户秤。很多方法被蘋果禁止或者漏洞太多码秉,越來(lái)越不被開發(fā)者使用,現(xiàn)在蘋果主推 IDFA 和 IDFV 這兩種方法鸡号,分別對(duì)外和對(duì)內(nèi)转砖,但是 IDFV 在 APP 重新安裝時(shí)會(huì)更改,所以我的方法是通過(guò)第一次生成的 IDFV 存儲(chǔ)到 keychain 中鲸伴,以后每次獲取標(biāo)識(shí)符都從 keychain 中獲取府蔗。
#import<UIKit/UIKit.h>
@interface MYVendorToll : NSObject
+ (NSString *)getIDFV;
@end
#import "MYVendorToll.h"
#import "MYKeyChainTool.h"
?@implementation MYVendorToll?
?+ (NSString *)getIDFV{?
NSString *IDFV = (NSString *)[MYKeyChainTool load:@"IDFV"];
? ? ? ? if ([IDFV isEqualToString:@""] || !IDFV) {?
? ? ? ? ? ? ? ? ?IDFV = [UIDevice currentDevice].identifierForVendor.UUIDString;??
? ? ? ? ? ? ? ? ?[MYKeyChainTool save:@"IDFV" data:IDFV];?
? ? ? ? }??
? ? ? ? ?return IDFV;
}?
@end
#import<Foundation/Foundation.h>
@interface MYKeyChainTool : NSObject
+ (void)save:(NSString *)service data:(id)data;
+ (id)load:(NSString *)service;
+ (void)deleteKeyData:(NSString *)service;
@end
#import "MYKeyChainTool.h"
@implementation MYKeyChainTool
+ (NSMutableDictionary *)getKeychainQuery:(NSString *)service {
return [NSMutableDictionary dictionaryWithObjectsAndKeys:
(id)kSecClassGenericPassword,(id)kSecClass,
service, (id)kSecAttrService,
service, (id)kSecAttrAccount,
(id)kSecAttrAccessibleAfterFirstUnlock,(id)kSecAttrAccessible,
nil];
}
+ (void)save:(NSString *)service data:(id)data {
//Get search dictionary
NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];
//Delete old item before add new item
SecItemDelete((CFDictionaryRef)keychainQuery);
//Add new object to search dictionary(Attention:the data format)
[keychainQuery setObject:[NSKeyedArchiver archivedDataWithRootObject:data] forKey:(id)kSecValueData];
//Add item to keychain with the search dictionary
SecItemAdd((CFDictionaryRef)keychainQuery, NULL);
}
+ (id)load:(NSString *)service {
id ret = nil;
NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];
//Configure the search setting
//Since in our simple case we are expecting only a single attribute to be returned (the password) we can set the attribute kSecReturnData to kCFBooleanTrue
[keychainQuery setObject:(id)kCFBooleanTrue forKey:(id)kSecReturnData];
[keychainQuery setObject:(id)kSecMatchLimitOne forKey:(id)kSecMatchLimit];
CFDataRef keyData = NULL;
if (SecItemCopyMatching((CFDictionaryRef)keychainQuery, (CFTypeRef *)&keyData) == noErr) {
@try {
ret = [NSKeyedUnarchiver unarchiveObjectWithData:(__bridge NSData *)keyData];
} @catch (NSException *e) {
NSLog(@"Unarchive of %@ failed: %@", service, e);
} @finally {
}
}
if (keyData)
CFRelease(keyData);
return ret;
}
+ (void)deleteKeyData:(NSString *)service {
NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];
SecItemDelete((CFDictionaryRef)keychainQuery);
}
@end