iOS设备唯一标识符解决方案
最近在公司的项目中有记录设备唯一标识符的需求,通过唯一标识符去识别设备的注册类别从而进行角色的切换,在这个过程中查找了一些资料,在此稍作总结,留下一些痕迹,当然能给有同类需求的伙伴提供一些思路也是对自己的一些鼓励.
下面进入正题 ,关于唯一标识符,目前有以下几种解决方案
- UDID
全称是Unique Device Identifier,它就是苹果IOS设备的唯一识别码,iOS2.0以后UIDevice提供一个获取设备唯一标识符的方法uniqueIdentifier,通过该方法可以获取设备的序列号,iOS5.0废弃,苹果公司认为可能泄露用户隐私 - UUID
UUID是Universally Unique Identifier的缩写,中文意思是通用唯一识别码。它是让分布式系统中的所有元素,都能有唯一的辨识资讯,而不需要透过中央控制端来做辨识资讯的指定。这样,每个人都可以建立不与其它人冲突的 UUID。苹果公司建议使用UUID为应用生成唯一标识字符串。 iOS 6中介绍的Vendor或Advertising标示符,通过下面的代码可以获得一个UUID字符串:
NSString *uuid = [[NSUUID UUID] UUIDString];
也可以保存在iOS 6中新增的Vindor标示符 (IDFV-identifierForVendor),获取这个IDFV的新方法被添加在已有的UIDevice类中。跟advertisingIdentifier一样,该方法返回的是一个NSUUID对象。
NSString *uuid = [[[UIDevice currentDevice] identifierForVendor] UUIDString];
开发者可以在应用第一次启动时调用一 次,然后将该串存储起来,以便以后替代UDID来使用。但是,如果用户删除该应用再次安装时,又会生成新的字符串,所以不能保证唯一识别该设备。由于IOS系统存储的数据都是在sandBox里面,一旦删除App,sandBox也不复存在。好在有一个例外,那就是keychain(钥匙串)。通常情况下,IOS系统用NSUserDefaults存储数据信息,但是对于一些私密信息,比如密码、证书等等,就需要使用更为安全的keychain了。
keychain里保存的信息不会因App被删除而丢失。所以,可以利用这个keychain这个特点来保存设备唯一标识。所以这里要用到KeyChain来保存。
那么,如何在应用里使用使用keyChain呢,我们可以使用已经封装好了的工具类KeychainItemWrapper来对keychain进行操作。
KeychainItemWrapper是apple官方例子“GenericKeychain”里一个访问keychain常用操作的封装类,在官网上下载了GenericKeychain项目后,只需要把“KeychainItemWrapper.h”和“KeychainItemWrapper.m”拷贝到我们项目,由于这两个文件是MRC的文件,在targets的build phases选项下Compile Sources下选择KeychainItemWrapper.m,双击它,输入-fno-objc-arc。
KeychainItemWrapper的用法:
NSString *idString = [OpenUDID value];
KeychainItemWrapper *wrapper = [[KeychainItemWrapper alloc] initWithIdentifier:@"cn.com.afcatrbx.openUdid--test" accessGroup:nil];
[wrapper setObject:idString forKey: (__bridge NSString *)kSecValueData];
在此代码示例中我使用的是OpenUDID
3.MAC地址
苹果推荐使用UUID 但是也有诸多问题,从而使用MAC地址。而MAC地址跟UDID一样,存在隐私问题,在iOS7上,如果请求Mac地址都会返回一个固定 值,那么Mac Address+bundle_id这个值大家的设备都变成一致的啦,跟UDID一样相当于被禁用
4.OpenUDID
OPEN UDID,没有用到MAC地址,同时能保证同一台设备上的不同应用使用同一个OpenUDID,只要用户设备上有一个使用了OpenUDID的应用存在时,其他后续安装的应用如果获取OpenUDID,都将会获得第一个应用生成的那个。但是根据贡献者的代码和方法,和一些开发者的经验,如果把使用了OpenUDID方案的应用全部都删除,再重新获取OpenUDID,此时的OpenUDID就跟以前的不一样,这一点已经在项目中验证过,目前我正在开发的两款APP一款是给银行的导服人员使用,另一款用来给办理业务的人员进行自助操作,需要对设备进行注册,但是如果两款App同时被删除,获取到的UDID会发生改变,原来的注册信息也就没有意义了.
5.推送token+bundle_id
推送token+bundle_id的方法:
1、应用中增加推送用来获取token
2、获取应用bundle_id
3、根据token+bundle_id进行散列运算
apple push token保证设备唯一,但必须有网络情况下才能工作,该方法不依赖于设备本身,但依赖于apple push,而苹果push有时候会抽风的。
我的解决方案是想采用OpenUDID + KeyChain进行解决,代码实例如下
//
// ViewController.m
// openUdid -test
//
// Created by sf on 16/6/10.
// Copyright © 2016年 Mr_Zhao. All rights reserved.
//
#import "ViewController.h"
#import "OpenUDID.h"
#import "KeychainItemWrapper.h"
@interface ViewController ()
@property (weak, nonatomic) IBOutlet UILabel *label;
@property (weak, nonatomic) IBOutlet UILabel *keyChain;
@property (nonatomic, strong) KeychainItemWrapper *wrapper;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSString *idString = [OpenUDID value];
KeychainItemWrapper *wrapper = [[KeychainItemWrapper alloc] initWithIdentifier:@"cn.com.afcatrbx.openUdid--test" accessGroup:nil];
self.wrapper = wrapper;
[wrapper setObject:idString forKey: (__bridge NSString *)kSecValueData];
self.label.text = idString;
}
-(void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event{
self.keyChain.text = [self.wrapper objectForKey: (__bridge NSString *)kSecValueData];
}
但是,在KeyChain中一个应用程序也只能读取到自己程序记录在KeyChain中的数据,保存在KeyChain的数据在同一个程序中是能够取到的,但是在另一个程序中读取不到.
6.集成华为AnyOffice SDK
在华为的AnyOffice中有获取设备标识的方法,但是具体他是怎么拿到的,我们看不到,亲测在装有AnyOffice SDK的应用全部被卸载后,再重新安装,通过它的方法拿到的deviceID 没有发生变化.这样可以解决我的问题,相信也能解决大部分开发者的获取设备唯一标识的开发需求
使用方式如下,下载AnyOffice SDK ,导入Svnsdk/MdmDeviceIdInfo.h
一句代码搞定
#import <Svnsdk/MdmDeviceIdInfo.h>
NSString *MDMUDID = [MdmDeviceIdInfo getDeviceID];
对于华为的AnyOfficeSDK为何能能拿到设备的UDID比较好奇,就开始研究怎样去获取UDID,各种搜索之后发现可以通过Safari去获取,这个已经有大神做过了:
感谢 天狐 为我们提供的这种方法!!!!:
原文链接: 通过Safari获取UDID
测试接口:获取设备UDID
一、通过苹果Safari浏览器获取iOS设备UDID步骤
苹果公司允许开发者通过IOS设备和Web服务器之间的某个操作,来获得IOS设备的UDID(包括其他的一些参数)。概述:
- 在你的Web服务器上创建一个.mobileconfig的XML格式的描述文件;
- 用户在所有操作之前必须通过某个点击操作完成.mobileconfig描述文件的安装(在使用华为AnyOffice的时候也会有这一步);
- 服务器需要的数据,比如:UDID,需要在.mobileconfig描述文件中配置好,以及服务器接收数据的URL地址;
- 当用户设备完成数据的手机后,返回提示给客户端用户;
二、.mobileconifg
在这篇文章中,主要讲如何获得标识符。其实还可以获取更多信息,以下是一个获得UDID示例.mobileconfig配置
/ConfigurationProfileExamples/ConfigurationProfileExamples.html-->
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>PayloadContent</key>
<dict>
<key>URL</key>
<string>http://dev.skyfox.org/udid/receive.php</string> <!--接收数据的接口地址-->
<key>DeviceAttributes</key>
<array>
<string>UDID</string>
<string>IMEI</string>
<string>ICCID</string>
<string>VERSION</string>
<string>PRODUCT</string>
</array>
</dict>
<key>PayloadOrganization</key>
<string>dev.skyfox.org</string> <!--组织名称-->
<key>PayloadDisplayName</key>
<string>查询设备UDID</string> <!--安装时显示的标题-->
<key>PayloadVersion</key>
<integer>1</integer>
<key>PayloadUUID</key>
<string>3C4DC7D2-E475-3375-489C-0BB8D737A653</string> <!--自己随机填写的唯一字符串-->
<key>PayloadIdentifier</key>
<string>dev.skyfox.profile-service</string>
<key>PayloadDescription</key>
<string>本文件仅用来获取设备ID</string> <!--描述-->
<key>PayloadType</key>
<string>Profile Service</string>
</dict>
</plist>
你需要填写回调数据的URL和PayloadUUID。该PayloadUUID仅仅是随机生成的唯一字符串,用来标识唯一
注意:mobileconfig下载时设置文件内容类型Content Type为:application/x-apple-aspen-config
三、iOS设备安装.mobileconfig描述文件
新建一个用于下载mobileconfig的网页,这里我命名为udid.php
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<meta content="width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0,user-scalable=no" name="viewport" id="viewport" />
<title>获取您的UDID</title>
<body>
<div id="content">
UUDI:<input style="" name="" value="$udid" />
<a class="buttons" href="udid.mobileconfig" target="_blank">1.点击获取您的UDID</a>
<a class="buttons" href="yourapp://?function=valid&uuid=$udid">2.验证ipa</a>
</div>
</body>
</html>
//yourapp为应用提前设置的URL Schemes(查看自定义 URL Scheme 完全指南)
下面的界面就是用户通过浏览器点击开始安装时的界面,用户点击“Install/安装”开始安装,下面的mobileconfig文件是没有签名的,所以会显示“Unsigned/未签名”红色提示,并且安装的时候还会多出一部警告界面;点击查看:为iOS的mobileconfig文件进行签名
四、服务器接收返回数据
设置好mobileconfig文件中的URL,并且下载安装mobileconfig之后,iOS设备会POST XML数据流给你的URL
receive.php
<?php $data = file_get_contents('php://input'); //这里可以进行xml解析 //header("Location: http://dev.skyfox.org/udid?data=".rawurlencode($data)); //有人说必须得目录形式才会安装成功 header('HTTP/1.1 301 Moved Permanently'); //这里一定要301跳转,否则设备安装会提示"无效的描述文件" header("Location: http://dev.skyfox.org/udid/index.php?".$params); ?>
java版本receive.do
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { response.setContentType("text/html;charset=UTF-8"); request.setCharacterEncoding("UTF-8"); //获取HTTP请求的输入流 InputStream is = request.getInputStream(); //已HTTP请求输入流建立一个BufferedReader对象 BufferedReader br = new BufferedReader(new InputStreamReader(is,"UTF-8")); StringBuilder sb = new StringBuilder(); //读取HTTP请求内容 String buffer = null; while ((buffer = br.readLine()) != null) { sb.append(buffer); } String content = sb.toString().substring(sb.toString().indexOf("<?xml"), sb.toString().indexOf("</plist>")+8); //content就是接收到的xml字符串 //进行xml解析即可 String udid = response.setStatus(301); //301之后iOS设备会自动打开safari浏览器 response.setHeader("Location", "http://192.168.1.106:8080/udid.jsp?UDID="+udid); //http://192.168.1.106:8080/udid.jsp 是用于显示udid的页面,也可以利用之前的下载mobileprofile文件页面 }
值得注意的是重定向一定要使用301重定向,有些重定向默认是302重定向,这样就会导致安装失败,设备安装会提示”无效的描述文件
以下是返回数据的格式
<!DOCTYPE plist PUBLIC "-//Apple//DTD PLIST 1.0//EN" "http://www.apple.com/DTDs/PropertyList-1.0.dtd">
<plist version="1.0">
<dict>
<key>IMEI</key>
<string>12 123456 123456 7</string>
<key>PRODUCT</key>
<string>iPhone8,1</string>
<key>UDID</key>
<string>b59769e6c28b73b1195009d4b21cXXXXXXXXXXXX</string>
<key>VERSION</key>
<string>15B206</string>
</dict>
</plist>
由于目前做的项目发布再企业应用商店可以通过这样的方式去获取,但是在获取之后还能否上架AppStore不是很确定
希望这篇文章能够帮到你
参考文章:iOS 获取设备唯一标示符的方法
华为AnySDK下载地址:华为AnySDK下载地址