由于安卓平台的开放性和碎片化,对于我们获取唯一可靠的设备标识符带来了很大的困难,在本片文章中我们将介绍安卓平台下的,各种获取设备标识符的方式以及这些方式的利弊,那什么是设备标识符呢?
简单说,设备标识符就是唯一标识该设备的一串代码或符号。
设备标识符在Android开发中的作用
- 跟踪用户安装
- 识别物理设备
获取Android设备ID(DeviceId)
设备ID(DeviceId)是手机唯一标识符,属于比较稳定的设备标识符。对于GSM返回IMEI,对于CDMA返回MEID或ESN。
//uses-permission android:name="android.permission.READ_PHONE_STATE"
private String getDeviceId() {
TelephonyManager telephonyManager = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
String deviceId = telephonyManager.getDeviceId();
return deviceId;
}
缺陷:
- 非手机设备没有这种唯一的标识符,比如平板电脑、机顶盒、音乐播放器等。
- 权限问题:调用方法需要READ_PHONE_STATE权限,特别是Android 6.0以后,权限需要用户手动确认。
- Bug:在一些设备制造商的设备上,这个只是不准确的 ,比如会返回一堆0或者星号。
获取Mac Address
通过检索设备的Wi-Fi或者蓝牙硬件是有可能取得设备的Mac地址的。
// uses-permission android:name="android.permission.ACCESS_WIFI_STATE"
private String getWifiMacDId() {
WifiManager wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);
String wifMacId = wifiManager.getConnectionInfo().getMacAddress();
return wifMacId;
}
//uses-permission android:name="android.permission.BLUETOOTH"
private String getBlueToothMacDId() {
BluetoothAdapter bluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
String blueToothMacId = bluetoothAdapter.getAddress();
return blueToothMacId;
}
缺陷:
- 硬件限制: 并不是所有的设备都有WiFi和蓝牙硬件
- 权限问题:需要ACCESS_WIFI_STATE或者BLUETOOTH权限
- 系统限制:为了保护用户隐私,从Android 6.0开始将不能获取WIFI和蓝牙有效地Mac地址(02:0:00:00:00:0)
SERIAL
从Android 2.3 开始,可以获取设备的硬件序列号
private String getSerialId(){
return android.os.Build.SERIAL;
}
缺陷:
Bug:可能不存在序列号的情况
Sim Serial Number 获取SIM卡的序列号
//uses-permission android:name="android.permission.READ_PHONE_STATE"
private String getSimSerialNumber() {
TelephonyManager telephonyManager = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
String simSerialNumber = telephonyManager.getSimSerialNumber();
return simSerialNumber;
}
缺陷:
- 设备限制:仅装有SIM卡的设备才能获取到
- 权限问题:调用方法需要READ_PHONE_STATE权限,特别是Android 6.0以后,权限需要用户手动确认。
- Bug :对于CDMA设备,返回的是一空值
Android ID
设备第一次启动时产生和存储的 64-bit 字符串
private String getAndroidId() {
return Settings.Secure.getString(getContentResolver(), Settings.Secure.ANDROID_ID);
}
缺陷:
- 恢复出厂设置后可能会重置该值。
- 部分设备由于制造商错误实现,导致返回相同的 Android_ID 。
- 在 Android 4.2 及以上,设备启用多用户功能后,每个用户的 Android ID 不同。
UUID
全局唯一标识符,是指在一台机器上生成的数字,它保证对在同一个时空中所有的机器都是唯一的
private String getUUID() {
return java.util.UUID.randomUUID().toString();
}
缺陷:
- 和设备无关,不同的应用会产生不同的ID,无法做到设备唯一
如何选择设备标识符
- 跟踪用户安装,不需要严格意义上设备唯一的ID,使用UUID即可跟踪用户的安装
- 识别物理设备,使用多个硬件标识符进行拼装
private String getUUID() {
return md5(androidId()+serialNumber()+" ");
}
识别模拟器
public static boolean runnigOnEmulator() {
if (!TextUtils.isEmpty(Build.MODEL) && Build.MODEL.toLowerCase().contains("sdk")) {
return true;
}
if (!TextUtils.isEmpty(Build.MANUFACTURER) && Build.MANUFACTURER.toLowerCase().contains("unknown")) {
return true;
}
if (!TextUtils.isEmpty(Build.DEVICE) && Build.DEVICE.toLowerCase().contains("generic")) {
return true;
}
return false;
}