欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

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

程序员文章站 2022-03-09 22:49:21
...

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

场景

最近在做一个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文档唯一标识符最佳做法你也许可以这样…


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);
    }
}
相关标签: Android