「转」识别独立安卓设备

通过生成唯一的识别码,识别独立的安卓手机的几种方法

本文转载自难题:识别独立安卓设备 - 「A WING BY WIND」

今天碰巧朋友问到我怎么识别独立的安卓手机,就花了一些时间琢磨了一下。其实这个问题可以秒答,就是 IMEI 。

1
TelephonyManager.getDeviceId();

这需要一个权限:

1
<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 可以这样查看:

1
adb shell getprop ro.serialno

代码中可使用系统变量 android.os.Build.SERIAL 访问。如果这个值能够取到这是仅次于 IMEI 的最好方法了。缺点是这个值在 2.2 及以下版本的 Android 系统不支持。不过好在如今的安卓世界 2.2 及以下的占有率已经越来越低了,翻新速度很快,因此这个值很值得一试。

3. Android ID

其实安卓系统提供了 Settings.Secure.ANDROID_ID 来获取唯一设备号。

1
2
3
4
5
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 都是可取的做法了。

6. 文章来源:

  1. 难题:识别独立安卓设备