一、简单的介绍下IDFV(identifierForVendor)
Vindor标示符,适用于对内:例如分析用户在应用内的行为等。是给Vendor
标识用户用的,每个设备在所属同一个Vender
的应用里,都有相同的值。其中的Vender
是指应用提供商(app所属的公司),但准确点说,是通过BundleID
的DNS
反转的前两部分进行匹配,如果相同就是同一个Vender
,例如对于com.somecompany.appone
和com.somecompany.apptwo
,这两个BundleID
来说,就属于同一个Vender
,共享同一个IDFV
的值。idfv的值是一定能取到的,所以非常适合于作为内部用户行为分析的主id
,来标识用户,替代OpenUDID
。
注意:如果用户将属于此Vender
的所有App
卸载,则IDFV的值会被重置,即再重装此Vender
的App
,IDFV的值和之前不同。
二、对IDFV数据变化的实践测试
- 1.获取IDFV
NSString * idfv = [[[UIDevice currentDevice] identifierForVendor] UUIDString];
- 2.在开发者账号里面没有套装ID的
Bundle identifier
工程里面测试
//保存上一次的IDFV NSString *oldIDFV = @"EA3D58A145B84F249C454A5A02E3BABD"; NSString *idfv = [[[UIDevice currentDevice] identifierForVendor] UUIDString]; idfv = [idfv stringByReplacingOccurrencesOfString:@"-" withString:@""]; NSLog(@"用户的IDFV=%@ 长度=%ld",idfv,idfv.length); if ([oldIDFV isEqualToString:idfv]) { NSLog(@"没有发生变化"); }else{ NSLog(@"有发生变化"); }
- 测试结果:同一个手机在没上线的不同工程里
IDFV
是一样的,不管app被卸载多少次,IDFV
是不变的。但是在上线的app工程里IDFV
是只要app卸载了再安装IDFV
就会发生变化。
- 3.在开发者账号里面有套装ID的
Bundle identifier
工程里面测试(在此提供一个:com.yibangbao.Ybangbao
)
还是用上面的代码来测试,只是把Bundle identifier
换成了有有套装ID的Bundle identifier
测试结果:只要app不卸载 IDFV
是不会变化的,只要卸载了再安装 IDFV
就会发生变化
三、对IDFV进行KeyChain存储在手机里
- 3.1、如果用户卸载了同一个vendor对应的所有程序,然后在重新安装同一个vendor提供的程序,此时identifierForVendor会被重置,所以这里要用到KeyChain来保存。
KeyChain(钥匙串)是使用苹果设备经常使用的,通常要调试的话,都得安装证书之类的,这些证书就是保存在KeyChain中,还有我们平时浏览网页记录的账号密码也都是记录在KeyChain中。iOS中的KeyChain相比OS X比较简单,整个系统只有一个KeyChain,每个程序都可以往KeyChain中记录数据,而且只能读取到自己程序记录在KeyChain中的数据。iOS中Security.framework框架提供了四个主要的方法来操作KeyChain:
这四个方法参数比较复杂,一旦传错就会导致操作KeyChain失败,文档中介绍的比较详细,大家可以查查官方文档。
SecItemCopyMatching(CFDictionaryRef query, CFTypeRef *result);//查询OSStatus SecItemAdd(CFDictionaryRef attributes, CFTypeRef *result); //添加OSStatus SecItemUpdate(CFDictionaryRef query, CFDictionaryRef attributesToUpdate);//更新KeyChain中的ItemOSStatus SecItemDelete(CFDictionaryRef query)//删除KeyChain中的ItemOSStatus
- 3.2、使用方法一: 苹果有一个类KeychainItemWrapper可以下载下来使用
- 使用
KeychainItemWrapper
的一个小demoJKOCKeyChainUUID你可以下载下来使用 - 使用的代码
JKKeyChainUUID *keychain = [[JKKeyChainUUID alloc] init]; NSString *keychainValue = [keychain readUDID]; NSLog(@"keychainValue = %@",keychainValue);
- 其中
JKKeyChainUUID
中的两个宏你可以按照自己意思去改
// 取出唯一标识符的键 #define KEY_IN_KEYCHAIN @"JKkeychain" // kSecClass 是你存数据是什么格式,这里是通用密码格式 #define JKKSecClass @"JKkeychainApp"
- 3.3、使用方法二:方法而就比较简单了,运用github上面用的比较多的一个三方SAMKeychain,使用量还是比较大的。用CocoaPods导入后,导入头文件
#import "SAMKeychain.h"
为了使用更方便,我写了个类JKKeyChainUUID
方便更好的使用,使用方法如下
NSString *keyChain = [JKKeyChainUUID jkGetDeviceId]; NSLog(@"keyChain=%@",keyChain);
- 在码云上我也放了一份demo:JKOCGitKeyChainUUID