白嫖党可直接下滑到最后 (手动滑稽) 绕开那些不太完美的办法
场景
最近在做一个OA考勤项目,里面有个需求是一个账号只允许绑定一台设备, 这就需要将设备唯一标识码与账号绑定在一起,咋一听 这还不So Easy啊,直接获取deviceID 不就好了。。。
问题
private String getSerial() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
return Build.getSerial();
} else {
return Build.SERIAL;
}
}
咋一操作也挺好。 可到了大面积的群体用户测试环节发现了几个问题。
- 需要获取手机READ_PHONE_STATE权限,这个权限涉及的信息还挺多。我们为了设备id有点大材小用
- 在努比亚和一加手机上有返回null 和 000000 ,应该是厂商在这方面做了特别的处理来保护用户隐私
- 在Android10 就已经完全获取不到,返回unknown
- 如果是双卡,其中一张卡是电信的,手机是小米的,那么你更换卡槽的位置,deviceId会从IMIE1 变为MEID。华为、荣耀等机型未发现。(当然也有解决办法获取 getDeviceId(0)的方式)
在Android 10以下版本虽然可以获取到IMEI码,但是需要在应用获取到了READ_PHONE_STATE权限,并且我们仍然没法保证所有的设备都能正常返回
简言之就是如果你的手机需要考虑Android9以上的设备,你就没办法通过获取系统提供的API完全的获取到设备唯一标识码,根据Android文档唯一标识符最佳做法你也许可以这样....
呃.... 这个特殊运营商许可 和这个 READ_PRIVILEGED_PHONE_STATE
好像都不是我们一般的安卓应用能获取的到的。
想要能Android9以上又能不受到刷机,格式化的影响。 就只能 使用移动安全联盟(MSA)提出的补充设备标识
是由移动安全联盟提出的,包含以下三个标识:
名称 | 说明 |
---|---|
OAID | 匿名设备标识符,最长64为,所有应用都获取到同一个ID,但是用户可关闭、可重置 |
AAID | 应用匿名设备标识符,最长64为,每个应用获取到各自的ID |
VAID | 开发者匿名设备标识符,最长64为,同一开发者不同应用获取到的一致 |
缺点就是:并不是所有的品牌商都兼容了, 而且需要导入他们对应的包 。 这部分可参考 Android 10获取设备标识方案探究 这篇文章
楼主的做法
androidId + 序列号+ 硬件信息(品牌+型号)
特点: 如果格式化或者刷机,这个设备ID会更改(这个也是合理的)。
完善:可以在第一次生成后放入到内部存储和外部存储,下次进入先从内部存储获取,如果没有在去外部存储比较(App卸载,内部存储就没了)。 如果在没有就生成一个
public class SystemUtils {
/**
* ANDROID_ID(恢复出厂+刷机会变) + 序列号(android 10会unknown/android 9需要设备权限)+品牌 +机型
* @return
*/
@SuppressLint("MissingPermission")
public static String getUniqueIdentificationCode(FragmentActivity context){
String androidId = Settings.System.getString(context.getContentResolver(), Settings.Secure.ANDROID_ID);
String uniqueCode ;
if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
/** 需要权限 且仅适用9.0。 10.0后又不能获取了*/
uniqueCode = androidId + Build.getSerial()+Build.BRAND+ Build.MODEL;
}else{
uniqueCode = androidId + Build.SERIAL+Build.BRAND+ Build.MODEL;
}
return toMD5(uniqueCode);
}
/**
* MD5加密 格式一致
*/
private static String toMD5(String text){
MessageDigest messageDigest = null;
try {
messageDigest = MessageDigest.getInstance("MD5");
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
}
byte[] digest = messageDigest.digest(text.getBytes());
StringBuilder sb = new StringBuilder();
for (int i = 0; i < digest.length; i++) {
int digestInt = digest[i] & 0xff;
//将10进制转化为较短的16进制
String hexString = Integer.toHexString(digestInt);
if (hexString.length() < 2) {
sb.append(0);
}
sb.append(hexString);
}
return sb.toString().substring(8,24);
}
}