android 获得设备唯一标识码(总有一款适合你)

白嫖党可直接下滑到最后 (手动滑稽) 绕开那些不太完美的办法

场景

最近在做一个OA考勤项目,里面有个需求是一个账号只允许绑定一台设备, 这就需要将设备唯一标识码与账号绑定在一起,咋一听 这还不So Easy啊,直接获取deviceID 不就好了。。。

问题

private String getSerial() {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        return Build.getSerial();
    } else {
        return Build.SERIAL;
    }
}

咋一操作也挺好。 可到了大面积的群体用户测试环节发现了几个问题。

  1. 需要获取手机READ_PHONE_STATE权限,这个权限涉及的信息还挺多。我们为了设备id有点大材小用
  2. 在努比亚和一加手机上有返回null 和 000000 ,应该是厂商在这方面做了特别的处理来保护用户隐私
  3. 在Android10 就已经完全获取不到,返回unknown
  4. 如果是双卡,其中一张卡是电信的,手机是小米的,那么你更换卡槽的位置,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);
    }
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章