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 }
测试结果演示:
Android手机:
Android平板:
参考:获取Android设备唯一标识码