Android中获取唯一的id

Android唯一设备ID现状

  1. 设备ID,简单来说就是一串符号(或者数字),映射现实中硬件设备。如果这些符号和设备是——
    对应的,可称之为“唯一设备ID(Unique Device ldentifier)”
  2. 不幸的是,对于Android平台而言,没有稳定的API可以让开发者获取到这样的设备ID。
  3. 开发者通常会遇到这样的困境:随着项目的演进,越来越多的地方需要用到设备ID;然而随着
    Android版本的升级,获取设备ID却越来越难了。
  4. 加上Android平台碎片化的问题,获取设备ID之路,可以说是步履维艰。
  5. 获取设备标识的API屈指可数,而且都或多或少有一些问题。

IMEI

  1. IMEI本该最理想的设备ID,具备唯一性,恢复出厂设置不会变化(真正的设备相关),可通过拨打
    *#06#查询手机的imei码。
  2. 在Android 9.O以后彻底禁止第三方应用获取设备的IMEI(即使申请了 READ_PHONE_STATE权
    限)。所以,如果是新APP,不建议用IMEI作为设备标识;
  3. 如果已经用IMEI作为标识,要赶紧做兼容工作了,尤其是做新设备标识和IMEI的映射。

MAC地址

  1. 大多android设备都有wif模块,因此,wifi模块的MAC地址就可以作为设备标识。基于隐私考虑,官
    方不建议获取
  2. 获取MAC地址也是越来越困难了,Android 6.o以后通过 WifiManager获取到的mac将是固定的:
    02:00:00:00:00:00
  3. 7.0之后读取/sys/class/net/wlan0/address也获取不到了(小米6)。
  4. 10.0后的地址也放弃了,不能读取mac地址

唯一Id实现方案

方案1:UUID+SharePreference(存取)

优点:数据唯一、不需要权限;
缺点:会随APP一起删除,即:重新安装APP,Deviceld值会改变(新UUID);

方案2:UUID+SD卡(存取)

优点:数据唯一、不随APP一起删除;
缺点:需要SD卡读写权限;防不住用户手动删除SD卡的文件

方案3:imei+android id+serial+硬件uuid(自生成)

不依赖随机的UUID,咱们根据硬件标识来创建唯一的数据;但其需要重新申请权限。

方案4:所有能得到的硬件信息,组成一个序列集

那些硬件适合硬件标识

Androidld:如:df176fbb152ddce,无需权限极个别设备获取不到数据或得到错误数据;
serial:如:LKX7N18328000931,无需权限极个别设备获取不到数据;
IME1:如:23b12e30ec8a2f17,需要权限;
Mac:如:6e:a5需要权限,高版本手机获得数据均为02:00(不可使用)
Build.BOARD如:BLA主板名称,无需权限同型号设备相同
Build.BRAND如:HUAWEI厂商名称,无需权限同型号设备相同
Build.HARDWARE 如:kirin970硬件名称无需权限同型号设备相同

工具类

public class DeviceIdUtil {
    public static String getDeviceId(Context context) {

        StringBuilder sbDeviceId = new StringBuilder();
//        imei null
        String imei = getIMEI(context);
  //        imei null

//        手机型号 +手机
        String androidID = getAndroidId(context);

//        serial  8.0 拿不到    8.0拿到了
//         唯一
        String serial = getSerial();
//        UUID  uuid----》
        String id = getDeviceUUID().replace("-", "");
//追加imei
        if (imei != null && imei.length() > 0) {
            sbDeviceId.append(imei);
            sbDeviceId.append("|");
        }
        //追加androidid
        if (androidID != null && androidID.length() > 0) {
            sbDeviceId.append(androidID);
            sbDeviceId.append("|");
        }
        //追加serial
        if (serial != null && serial.length() > 0) {
            sbDeviceId.append(serial);
            sbDeviceId.append("|");
        }
        //追加硬件uuid
        if (id != null && id.length() > 0) {
            sbDeviceId.append(id);
        }
//        一系列的字符串  ----11 硬件标识有关   手机
        //生成SHA1,统一DeviceId长度
        if (sbDeviceId.length() > 0) {
//                    md  ----
            try {
                byte[] hash = getHashByString(sbDeviceId.toString());
                String sha1 = bytesToHex(hash);
                if (sha1 != null && sha1.length() > 0) {
                    //返回最终的DeviceId
                    return sha1;
                }
            } catch (Exception ex) {
                ex.printStackTrace();
            }

        }
        return null;

    }
    /**
     * 转16进制字符串
     *
     * @param data 数据
     * @return 16进制字符串
     */
    private static String bytesToHex(byte[] data) {
        StringBuilder sb = new StringBuilder();
        String stmp;
        for (int n = 0; n < data.length; n++) {
            stmp = (Integer.toHexString(data[n] & 0xFF));
            if (stmp.length() == 1)
                sb.append("0");
            sb.append(stmp);
        }
        return sb.toString().toUpperCase(Locale.CHINA);
    }
    /**
     * 取SHA1
     *
     * @param data 数据
     * @return 对应的hash值
     */
    private static byte[] getHashByString(String data) {
        try {
            MessageDigest messageDigest = MessageDigest.getInstance("SHA1");
            messageDigest.reset();
            messageDigest.update(data.getBytes("UTF-8"));
            return messageDigest.digest();
        } catch (Exception e) {
            return "".getBytes();
        }
    }

    // //获得硬件uuid(根据硬件相关属性,生成uuid)(无需权限)  数字  0   -10
    private static String getDeviceUUID() {
        String dev="100001"+Build.BOARD+
                Build.BRAND +
                Build.DEVICE +
                Build.HARDWARE +
                Build.ID +
                Build.MODEL +
                Build.PRODUCT +
                Build.SERIAL ;
        return new UUID(dev.hashCode(), Build.SERIAL.hashCode()).toString();
    }

    private static String getSerial() {
        try {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
                return Build.getSerial();
            }
        } catch (Exception ex) {
            ex.printStackTrace();
        }

        return null;
    }

    /**
     * 获得设备的AndroidId
     *
     * @param context 上下文
     * @return 设备的AndroidId
     */
    private static String getAndroidId(Context context) {
        try {
            return Settings.Secure.getString(context.getContentResolver(),
                    Settings.Secure.ANDROID_ID);
        } catch (Exception ex) {
            ex.printStackTrace();
        }
        return "";
    }
    //需要获得READ_PHONE_STATE权限,>=6.0,默认返回null
    private static String getIMEI(Context context) {
        try {
            TelephonyManager tm = (TelephonyManager)
                    context.getSystemService(Context.TELEPHONY_SERVICE);
            return tm.getDeviceId();
        } catch (Exception ex) {
            ex.printStackTrace();
        }
        return "";
    }
}
  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

陈德山

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值