Android 唯一标识

唯一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 "";
  }
}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,843评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,538评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,187评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,264评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,289评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,231评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,116评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,945评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,367评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,581评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,754评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,458评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,068评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,692评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,842评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,797评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,654评论 2 354

推荐阅读更多精彩内容