「博客搬家」 原地址: CSDN 原发表时间: 2016-04-15
今天碰巧朋友问到我怎么识别独立的安卓手机,就花了一些时间琢磨了一下。其实这个问题可以秒答,就是 IMEI 。
TelephonyManager.getDeviceId();
这需要一个权限:
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
问题如果这么简单就好了,问题在于:
- 平板等其他设备,无 SIM 卡,因此无 Telephony 模块。
- 有些设备恢复出厂设置的操作会清空 IMEI「水货硬解通常会导致这个结果」
- 我国许多山寨手机厂商偷懒,刷机不改 IMEI,所以 IMEI 都是一样的
因此其实问题从这里才开始。网上能够搜索到的解决方案有以下几点:
1. WIFI MAC
一个方案是优先采用 IMEI,当 IMEI 相同时,再比较 WIFI 的 MAC 地址。但如果手机没有 WIFI 功能或者 WIFI 功能没有开启「飞行模式」,则无法获取到 MAC 地址。更加让人惆怅的是,我国大山寨厂商实在是懒透了,无线网卡的 MAC 地址居然也不修改,不少自刷机的也是这病情「例如这个,这个还有这个」。至于说蓝牙 MAC 地址就更别说了。IMEI 重复的病因,与 MAC 地址相同的其实是一个原因,都是刷机或山寨,所以这「WIFI MAC」地址的方案其实算不上互补了,必须另谋途径。
2. Serial NO
另一个方案是用 serial NO 。这个值仅在 Android 2.3 版本以上才提供支持。通过 adb 可以这样查看:
adb shell getprop ro.serialno
代码中可使用系统变量 android.os.Build.SERIAL
访问。如果这个值能够取到这是仅次于 IMEI 的最好方法了。缺点是这个值在 2.2 及以下版本的 Android 系统不支持。不过好在如今的安卓世界 2.2 及以下的占有率已经越来越低了,翻新速度很快,因此这个值很值得一试。
3. Android ID
其实安卓系统提供了 Settings.Secure.ANDROID_ID
来获取唯一设备号。
import android.provider.Settings.Secure;
private String android_id = Secure.getString(
getContext().getContentResolver(),
Secure.ANDROID_ID);
但同样 2.2 以前的系统支持得不好。这个值是系统初次启动后生成的,因此恢复出厂设置后这个值会变,导致观测到的设备数虚高。在手机刷机、重置频繁的环境,这个值是不靠谱的。另外某大厂生产的设备居然有个 BUG,这个值是会重复的「DROID2」,因此这个值还是别用为好。
4. Generated UUID
最后还有一个自己生成 UUID 的办法,保存这个 ID,每次访问服务器时上传,自己告诉系统自己是谁。这个方法比所有硬件方法都更不靠谱,因为只需卸载软件和清理数据就会导致这个值被删除,从而产生新的 UUID,造成观测到的设备数量虚高。更别说刷机和恢复出厂设置了。这个方法非常适合用来统计软件安装次数,而非独立设备数。
5. 结论
综上所述,目前为止我还没找到非常完美的统计独立设备的方案,尤其是在中国这个水货、硬解、山寨、刷机市场泛滥和不规范的安卓世界,更是难上加难。
但是反过来想,是不是一台完全刷新,安装了全新的 ROM 的手机,就已经不是原来的那台手机了呢?统计独立设备数的目的究竟是什么?如果要的是独立活跃设备数,其实用自己生成的 UUID 已经足够,因为原来使用的那个 UUID 已经失去活性,可以忽略了。对于 APP 开发者而言这个情况与用户换了一台手机,完全弃用旧手机的情况其实是一样的。假如是采取活跃 UUID 的方式,则即使是使用 ANDROID_ID 或者自己生成的 UUID 都是可取的做法了。