[iOS 开发] 如何获取设备唯一标识?以及多个App间如何共享同一个设备唯一标识?

来源:互联网 发布:ubuntu删除文件 编辑:程序博客网 时间:2024/06/04 20:11

更新2016.12.31:最近发现项目中用的OpenUDID 会导致应用崩溃,已更换为 FCUUID,需要注意的是,原来 OpenUDID 记录的设备 id (40位)是不能直接迁移到FCUUID 的,尽管 FCUUID 提供了这样的接口,因为 OpenUDID 保存的设备 id 并不是直接来自系统的UUID,而是经过OpenUDID 处理过的,所以不符合 FCUUID 中对 UUID 校验的规则(32位或者36位)。因此如果你想继续使用原来的设备 id,就需要自己使用存到 KeyChain 里面了。

最近公司项目有一个针对新用户的需求,对新用户的筛选条件是新账号&&新设备?YES:NO。烦人的问题就是这个对于新设备的判断了,相比轻松就能搞定的android开发者来说,iOS开发者真是有苦难言,谁让苹果公司真是时刻不忘一切为了用户啊——从2013年5月1日起,你再想直接拿用户设备的UDID?没门!
获取设备唯一标识符本来就不易,但是考虑到我司的两款app有可能同时上新用户活动,那就还得想办法保证同一设备上两款不同app获取到的标识是相同的,我滴个亲娘啊……

那怎么办呢?当然百度啊!找来找去,找到两个解决方案:OpenUDID框架和FCUUID框架。这两个框架都是采用“一次性创建CFUUID+持久性存储”的策略来实现山寨版的UDID的。(ps:这些造轮子的人真是牛x!感谢他们为我们广大码农提供便利!)

  • OpenUDID

问题一:现在还能用OpenUDID吗?
OpenUDID的作者已经在github上注明了[OpenUDID IS NOW DEPRECATED],时间是June 7 2013,并且作者还特别提醒我们在使用之前好好读一下这篇博客UDID is dead, OpenUDID is deprecated, long live advertisingIdentifier!。究竟为甚么被废弃了,我也是一头雾水,我司两款产品都一直在使用OpenUDID来作埋点统计,而且我自己也用真机测试过了,并没有发现什么未知异常。对此,github上也有人在2014年时提出了issue-Why deprecate OpenUDID ? #59,but...,没有人给出合理的解释。从github上的issue来看,使用者碰到的问题不少。知乎上一位使用OPenUDID的网友也声称遇到过不同设备的OpenUDID值相同的问题:不同设备的OpenUDID有可能重复吗?

问题二:怎么用OpenUDID来实现我们的需求?
OpenUDID的原理是用UIPasteboard来保存CFUUID,在iOS7以前,所有应用都可以访问同一个剪切板上存储的内容,但是iOS7以后只允许使用相同team ID的apps才能共享同一个剪切板上的数据。OpenUDID采用+ pasteboardWithName: create:方法来获取剪切板中的值,根据苹果的官方文档,可以得知,该方法返回的pasteboard可以用来在一个app内部或者两个拥有相同的team ID的app之间传递数据。
我用自己的个人证书创建了两个不同的app,这时就能满足两个不同的app拥有相同的team ID的条件了,连上iPhone5s分别运行,结果显示的是一样的值,说明OpenUDID是暂时可行的。

UIPasteboard官方文档.png
  • FCUUID
    问题一:FCUUID框架能满足需求吗?
    该框架诞生于2015年10月左右,其实现原理就是CFUUID+KeyChain,跟OpenUDID是差不多的,只是存储方式不同,需要注意的是,FCUUID框架依赖于框架UICKeyChainStore,。
    作者在github上明确说明了使用该框架的条件(如图所示),其中一条是:如果你想在不同的app间共享keychain中的数据,就需要保证这两个app有相同的bundle seed。那么什么是bundle seed呢?打开钥匙串,找到你的证书,bundle seed就是你证书名后面括号内的的10位编码。所以呢,有相同的bundle seed就意味着是同一个证书发布的应用。
FCUUID.png
Bundle seed.png

问题二:如何开启keychain sharing?
参考这篇文章iOS Keychain: Sharing data between apps(实际操作步骤不完全相同)。首先保证两个app都是由同一证书创建的-->点击capabilities-->打开keychain sharing-->Xcode会自动帮你生成entitlements文件-->分别在两个不同的项目中的keychain sharing中添加对方的bundle identifier-->显示两个勾选状态-->大功告成。通过真机测试,证实该方案可行。

综上,实际上获取“山寨UDID”的关键在于存储。而想要实现应用间共享数据的最关键之处在于是否是同一证书下管理的app。


参考文献

1.iOS唯一标示符引导:
http://www.cocoachina.com/ios/20130422/6040.html

2.iOS平台UDID方案比较:
http://www.cocoachina.com/ios/20130715/6597.html

3.KeyChain保存和获取应用的UDID:
http://blog.csdn.net/teng_ontheway/article/details/50200491

4.iOS7时代我们用什么来追踪和识别用户:
http://www.cnblogs.com/BigPolarBear/p/3359526.html

5.iOS的UDID废用以及UUID配合keychain的替换方案实现:
http://blog.k-res.net/archives/1081.html

6.iOS设备唯一标识探讨:
http://www.jianshu.com/p/b83b0240bd0e

相关开源框架
1.OpenUDID(简单易用,但在iOS7以后有局限性,只做参考):
https://github.com/ylechelle/OpenUDID

2.FCUUID(功能强大,但需设置iCloud / Key-value storage和KeyChain sharing):
https://github.com/fabiocaccamo/FCUUID