美文网首页
Android 唯一标识

Android 唯一标识

作者: 沪漂意哥哥 | 来源:发表于2022-05-31 20:22 被阅读0次

    唯一ID的重要性

    • 后台做大数据统计,为每个用户勾勒画像,需要唯一设备
    • 防止多设备重复登录,如QQ,微信
    • 某些付费功能,用户会通过卸载,重装 来达到一直免费使用
    • 在安全领域 格外重要,纪录那些设备登录过

    Android唯一设备ID现状

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

    唯一ID实现方式

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

    • 设备序列号

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

    解决方案

    • 方案1:UUID + SharePreference(存取)

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

    • 方案3:imei + android_id + serial + 硬件uuid(自生成)

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

    硬件标识

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

    代码工具

    import android.content.Context;
    import android.os.Build;
    import android.provider.Settings;
    import android.telephony.TelephonyManager;
    
    import java.security.MessageDigest;
    import java.util.Locale;
    import java.util.UUID;
    
    public class DeviceIdUtil {
      public static String getDeviceId(Context context) {
    
          StringBuilder sbDeviceId = new StringBuilder();
    
          String imei = getIMEI(context);
    //        手机型号 +手机
          String androidID = getAndroidId(context);
    
          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);
          }
          //生成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)(无需权限)
      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 "";
      }
    }
    
    

    相关文章

      网友评论

          本文标题:Android 唯一标识

          本文链接:https://www.haomeiwen.com/subject/ypftmrtx.html