zoukankan      html  css  js  c++  java
  • 20150919_获取Android唯一标识码

    背景

      前段时间给一家电力公司做了一个管理系统,用来调查公司客户的购买电力公司培训课程的意愿,并且提供下单订购的功能。

      因为电力公司要求在Android平板电脑上运行,所以是针对7英寸的Android平板电脑的开发;苦于没有合适的设备调试,所以我一直在自己的Android手机上进行测试,一直到项目提交之前的那一天晚上...
      客户反馈说提交订单后系统停止运行,导入数据到PC服务端后服务端也崩溃了!
      刚开始怀疑是因为他们的设备太差的缘故——我在配置相当low的Android原生虚拟机(有多low大家应该清楚吧-_-)上进行调试时都未曾出现崩溃的问题,为什么在一家国企的平板设备上就会跑不动呢?
      无奈让客户寄过来他们的设备进行实机调试,结果发现竟然在一处Log报出了空异常:

    1 TelephonyManager tm = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
    2 String DEVICE_ID = tm.getDeviceId();
    3 Log.v("DEVICE_ID = ", DEVICE_ID); // 异常

      原来,最开始搭载Android系统都是手机设备,而现在也出现了非手机设备:如平板电脑、电子书、电视、音乐播放器等。这些设备没有通话的硬件功能,系统中也就没有TELEPHONY_SERVICE,自然也就无法通过上面的方法获得DEVICE_ID。所有DEVICE_ID的值就为null了,这使得数据库中出现了空值,而服务端和客户端共享数据库,于是服务端也挂了。这样一来一切都说得通了。

    概述

      有时需要对用户设备进行标识,所以希望能够得到一个稳定可靠并且唯一的识别码。虽然Android系统中提供了这样设备识别码,但是由于Android系统版本、厂商定制系统中的Bug等限制,稳定性和唯一性并不理想。而通过其他硬件信息标识也因为系统版本、手机硬件等限制存在不同程度的问题。
    下面收集了一些“有能力”或“有一定能力”作为设备标识的串码。

    DEVICE_ID

      这是Android系统为开发者提供的用于标识手机设备的串号,也是各种方法中普适性较高的,可以说几乎所有的设备都可以返回这个串号,并且唯一性良好。

    这个DEVICE_ID可以同通过下面的方法获取:

    TelephonyManager tm = (TelephonyManager)getSystemService(Context.TELEPHONY_SERVICE); String DEVICE_ID = tm.getDeviceId(); 

      它会根据不同的手机设备返回IMEI,MEID或者ESN码,但在使用的过程中有以下问题:

    • 非手机设备:最开始搭载Android系统都手机设备,而现在也出现了非手机设备:如平板电脑、电子书、电视、音乐播放器等。这些设备没有通话的硬件功能,系统中也就没有TELEPHONY_SERVICE,自然也就无法通过上面的方法获得DEVICE_ID。
    • 权限问题:获取DEVICE_ID需要READ_PHONE_STATE权限,如果只是为了获取DEVICE_ID而没有用到其他的通话功能,申请这个权限一来大才小用,二来部分用户会怀疑软件的安全性。
    • 厂商定制系统中的Bug:少数手机设备上,由于该实现有漏洞,会返回垃圾,如:zeros或者asterisks

    MAC ADDRESS

      可以使用手机Wifi或蓝牙的MAC地址作为设备标识,但是并不推荐这么做,原因有以下两点:

    • 硬件限制:并不是所有的设备都有Wifi和蓝牙硬件,硬件不存在自然也就得不到这一信息。
    • 获取的限制:如果Wifi没有打开过,是无法获取其Mac地址的;而蓝牙是只有在打开的时候才能获取到其Mac地址。

      获取Wifi Mac地址:

    权限:<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"></uses-permission>
    	WifiManager wifi = (WifiManager) getSystemService(Context.WIFI_SERVICE); 
    	WifiInfo info = wifi.getConnectionInfo(); 
    	String Wifi_Mac = info.getMacAddress();

    Sim Serial Number

      装有SIM卡的设备,可以通过下面的方法获取到Sim Serial Number:

    TelephonyManager tm = (TelephonyManager)getSystemService(Context.TELEPHONY_SERVICE);
    String SimSerialNumber = tm.getSimSerialNumber(); 

    注意:对于CDMA(包括平板电脑这些没有sim卡的设备)设备,返回的是一个空值!

    ANDROID_ID

      在设备首次启动时,系统会随机生成一个64位的数字,并把这个数字以16进制字符串的形式保存下来,这个16进制的字符串就是ANDROID_ID,当设备被wipe后该值会被重置。可以通过下面的方法获取:

    import android.provider.Settings;   
    String ANDROID_ID = Settings.System.getString(getContentResolver(), Settings.System.ANDROID_ID); 

    ANDROID_ID可以作为设备标识,但需要注意:

    • 厂商定制系统的Bug:不同的设备可能会产生相同的ANDROID_ID:9774d56d682e549c。
    • 厂商定制系统的Bug:有些设备返回的值为null。
    • 设备差异:对于CDMA设备,ANDROID_ID和TelephonyManager.getDeviceId() 返回相同的值。

    Serial Number

      Android系统2.3版本以上可以通过下面的方法得到Serial Number,且非手机设备也可以通过该接口获取。

    String SerialNumber = android.os.Build.SERIAL; 

    Installtion ID

      以上几种方式都或多或少存在一定的局限性或者Bug,如果并不是确实需要对硬件本身进行绑定,使用自己生成的UUID也是一个不错的选择,因为该方法无需访问设备的资源,也跟设备类型无关。

      这种方式的原理是在程序安装后第一次运行时生成一个ID,该方式和设备唯一标识不一样,不同的应用程序会产生不同的ID,同一个程序重新安装也会不同。所以这不是设备的唯一ID,但是可以保证每个用户的ID是不同的。可以说是用来标识每一份应用程序的唯一ID(即Installtion ID),可以用来跟踪应用的安装数量等。

    Google Developer Blog提供了这样的一个框架:

     1 import java.io.File;
     2     import java.io.FileOutputStream;
     3     import java.io.IOException;
     4     import java.io.RandomAccessFile;
     5     import java.util.UUID;
     6 
     7     import android.content.Context;
     8     /**
     9      * Google Developer Blog,提供了这样的一个框架,用于获取设备的唯一标识
    10      * @author NULL1943
    11      *
    12      */
    13     public class TestInstallation {
    14         private static String sID = null;
    15         private static final String INSTALLATION = "INSTALLATION";
    16 
    17         public synchronized static String id(Context context) {
    18             if (sID == null) {
    19                 File installation = new File(context.getFilesDir(), INSTALLATION);
    20                 try {
    21                     if (!installation.exists())
    22                         writeInstallationFile(installation);
    23                     sID = readInstallationFile(installation);
    24                 } catch (Exception e) {
    25                     throw new RuntimeException(e);
    26                 }
    27             }
    28             return sID;
    29         }
    30 
    31         private static String readInstallationFile(File installation)
    32                 throws IOException {
    33             RandomAccessFile f = new RandomAccessFile(installation, "r");
    34             byte[] bytes = new byte[(int) f.length()];
    35             f.readFully(bytes);
    36             f.close();
    37             return new String(bytes);
    38         }
    39 
    40         private static void writeInstallationFile(File installation)
    41                 throws IOException {
    42             FileOutputStream out = new FileOutputStream(installation);
    43             String id = UUID.randomUUID().toString();
    44             out.write(id.getBytes());
    45             out.close();
    46         }
    47     }
    Installation ID获取框架

    测试结果演示:

    Android手机:

    Android平板:


    参考:获取Android设备唯一标识码
  • 相关阅读:
    在ACCESS中LIKE的用法
    pip 在windows下的更新升级
    NAS、SAN、DAS 说明
    RAID 工作模式
    Linux mail 邮件发送
    Linux 邮件服务搭建
    HA 脑裂原理
    Tomcat 工作原理
    Nagios 工作原理
    Nginx 工作原理
  • 原文地址:https://www.cnblogs.com/wanlong/p/wanlong_android_20150919001.html
Copyright © 2011-2022 走看看