Android系统以及设备都有很多的“标识”号,比如常见的IMEI,SerizalNumber,UUID等概念,但是这些都存在一定程度上的不可靠性,到底如何标记一台Android设备?
文章内容多来自:
1)http://www.cnblogs.com/lvcha/p/3721091.html
2)http://android-developers.blogspot.com/2011/03/identifying-app-installations.html
尤其第二篇,乃是网上许多文章的根源所在,第一篇文章则是在2)上得补充,整理得本文。
1. DEVICE_ID
假设我们确实需要用到真实设备的标识,可能就需要用到DEVICE_ID。在以前,我们的Android设备是手机,这个DEVICE_ID可以同通过TelephonyManager.getDeviceId()获取,它根据不同的手机设备返回IMEI,MEID或者ESN码,但它在使用的过程中会遇到很多问题:
1)非手机设备: 如果只带有Wifi的设备或者音乐播放器没有通话的硬件功能的话就没有这个DEVICE_ID
2)权限: 获取DEVICE_ID需要READ_PHONE_STATE权限,但如果我们只为了获取它,没有用到其他的通话功能,那这个权限有点大才小用
3)bug:在少数的一些手机设备上,该实现有漏洞,会返回垃圾,如:zeros(全是0)或者asterisks(星号)的产品
2. MAC ADDRESS
我们也可以通过手机的Wifi或者蓝牙设备获取MAC ADDRESS作为DEVICE ID,但是并不建议这么做,因为:
1)硬件限制:并不是所有的设备都有Wifi和蓝牙硬件,硬件不存在自然也就得不到这一信息。
2)获取的限制:如果Wifi没有打开过,是无法获取其Mac地址的;而蓝牙是只有在打开的时候才能获取到其Mac地址。
3. Serial Number
在Android 2.3可以通过android.os.Build.SERIAL获取,非手机设备可以通过该接口获取。没有电话功能的设备被要求必须提供这样一个序列号。
4. ANDROID_ID
ANDROID_ID是设备第一次启动时产生和存储的64bit的一个数,当设备被wipe后该数重置ANDROID_ID似乎是获取Device ID的一个好选择,但它也有缺陷:
1)它在Android <=2.1 or Android >=2.3的版本是可靠、稳定的,但在2.2的版本并不是100%可靠的
2)在主流厂商生产的设备上,有一个很经常的bug,就是每个设备都会产生相同的ANDROID_ID:9774d56d682e549c
3)厂商定制系统的Bug:有些设备返回的值为null。
4)设备差异:对于CDMA设备,ANDROID_ID和TelephonyManager.getDeviceId() 返回相同的值。
5. Installtion ID : UUID
以上四种方式都有或多或少存在的一定的局限性或者bug,在这里,有另外一种方式解决,就是使用UUID,该方法无需访问设备的资源,也跟设备类型无关。
这种方式是通过在程序安装后第一次运行后生成一个ID实现的(注意:是自己生成,而非系统主动生成的!),但该方式跟设备唯一标识不一样,它会因为不同的应用程序而产生不同的ID,而不是设备唯一ID。因此经常用来标识在某个应用中的唯一ID(即Installtion ID),或者跟踪应用的安装数量。很幸运的,Google Developer Blog提供了这样的一个框架:
1 public class Installation { 2 private static String sID = null; 3 private static final String INSTALLATION = "INSTALLATION"; 4 5 public synchronized static String id(Context context) { 6 if (sID == null) { 7 File installation = new File(context.getFilesDir(), INSTALLATION); 8 try { 9 if (!installation.exists()) 10 writeInstallationFile(installation); 11 sID = readInstallationFile(installation); 12 } catch (Exception e) { 13 throw new RuntimeException(e); 14 } 15 } 16 return sID; 17 } 18 19 private static String readInstallationFile(File installation) throws IOException { 20 RandomAccessFile f = new RandomAccessFile(installation, "r"); 21 byte[] bytes = new byte[(int) f.length()]; 22 f.readFully(bytes); 23 f.close(); 24 return new String(bytes); 25 } 26 27 private static void writeInstallationFile(File installation) throws IOException { 28 FileOutputStream out = new FileOutputStream(installation); 29 String id = UUID.randomUUID().toString(); 30 out.write(id.getBytes()); 31 out.close(); 32 } 33 }