Android手机唯一标识

有很多场景和需求需要用到手机设备的唯一标识符来使你的软件运行在指定的手机上。

在Android中,有以下几种方法获取这样的ID。


1、The IMEI:仅仅只对Android手机有效。

TelephonyManager telephonyMgr = (TelephonyManager)getSystemService(TELEPHONY_SERVICE); 

String m_imei = telephonyMgr.getDeviceId(); 

采用此种方法,需要在AndroidManifest.xml中加入一个许可:android.permission.READ_PHONE_STATE,并且用户应当允许安装此应用。作为手机来讲,IMEI是唯一的,它应该类似于 359881030314356(除非你有一个没有量产的手机(水货)它可能有无效的IMEI,如:0000000000000)。

2、Pseudo-Unique ID:这个在任何Android手机中都有效。

String m_devIDShort = "35" + 
Build.BOARD.length()%10+ Build.BRAND.length()%10 + Build.CPU_ABI.length()%10 + Build.DEVICE.length()%10 +Build.DISPLAY.length()%10 +Build.HOST.length()%10 + Build.ID.length()%10 + Build.MANUFACTURER.length()%10 +Build.MODEL.length()%10 + Build.PRODUCT.length()%10 +Build.TAGS.length()%10 + Build.TYPE.length()%10 +Build.USER.length()%10 ; 

有一些特殊的情况,一些如平板电脑的设置没有通话功能,或者你不愿加入READ_PHONE_STATE许可。而你仍然想获得唯一序列号之类的东西。这时你可以通过取出ROM版本、制造商、CPU型号、以及其他硬件信息来实现这一点。这样计算出来的ID不是唯一的(因为如果两个手机应用了同样的硬件以及Rom 镜像)。但应当明白的是,出现类似情况的可能性基本可以忽略。大多数的Build成员都是字符串形式的,我们只取他们的长度信息。我们取到13个数字,并在前面加上“35”。这样这个ID看起来就和15位IMEI一样了。

3、The Android ID 

String m_androidID = Secure.getString(getContentResolver(), Secure.ANDROID_ID);

通常被认为不可信,因为它有时为null。开发文档中说明了:这个ID会改变如果进行了出厂设置。并且,如果某个Andorid手机被Root过的话,这个ID也可以被任意改变。Returns: 9774d56d682e549c 。无需任何许可。

4、The WLAN MAC Address string:是另一个唯一ID。

WifiManager wm = (WifiManager)getSystemService(Context.WIFI_SERVICE); 

String m_wlanMac = wm.getConnectionInfo().getMacAddress();

你需要为你的工程加入android.permission.ACCESS_WIFI_STATE 权限,否则这个地址会为null。Returns: 00:11:22:33:44:55 (这不是一个真实的地址。而且这个地址能轻易地被伪造。)。WLan不必打开,就可读取些值。

5、The BT MAC Address string:只在有蓝牙的设备上运行。
BluetoothAdapter m_bluetoothAdapter = null; 
m_bluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); 
String m_bluetooth = m_bluetoothAdapter.getAddress();

你需要为你的工程加入android.permission.BLUETOOTH 权限。Returns: 43:25:78:50:93:38 。蓝牙没有必要打开,也能读取。


综上所述,我们一共有五种方式取得设备的唯一标识。它们中的一些可能会返回null,或者由于硬件缺失、权限问题等获取失败。但你总能获得至少一个能用。所以,最好的方法就是通过拼接,或者拼接后的计算出的MD5值来产生一个结果。

TelephonyManager telephonyMgr = (TelephonyManager) getSystemService(TELEPHONY_SERVICE);
		String m_imei = telephonyMgr.getDeviceId();

		String m_devIDShort = "35" + Build.BOARD.length() % 10 + Build.BRAND.length() % 10 + Build.CPU_ABI.length() % 10
				+ Build.DEVICE.length() % 10 + Build.DISPLAY.length() % 10 + Build.HOST.length() % 10
				+ Build.ID.length() % 10 + Build.MANUFACTURER.length() % 10 + Build.MODEL.length() % 10
				+ Build.PRODUCT.length() % 10 + Build.TAGS.length() % 10 + Build.TYPE.length() % 10
				+ Build.USER.length() % 10;

		String m_androidID = Secure.getString(getContentResolver(), Secure.ANDROID_ID);

		WifiManager wm = null;
		wm = (WifiManager) getSystemService(Context.WIFI_SERVICE);
		String m_wlanMac = wm.getConnectionInfo().getMacAddress();

		BluetoothAdapter m_bluetoothAdapter = null;
		m_bluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
		String m_bluetooth = m_bluetoothAdapter.getAddress();

		String m_longID = m_imei + m_devIDShort + m_androidID + m_wlanMac + m_bluetooth;
		// compute md5
		MessageDigest m = null;
		try {
			m = MessageDigest.getInstance("MD5");
		} catch (NoSuchAlgorithmException e) {
			e.printStackTrace();
		}
		m.update(m_longID.getBytes(), 0, m_longID.length());
		// get md5 bytes
		byte p_md5Data[] = m.digest();
		// create a hex string
		String m_uniqueID = new String();
		for (int i = 0; i < p_md5Data.length; i++) {
			int b = (0xFF & p_md5Data[i]);
			// if it is a single digit, make sure it have 0 in front (proper
			// padding)
			if (b <= 0xF)
				m_uniqueID += "0";
			// add number to string
			m_uniqueID += Integer.toHexString(b);
		} // hex string to uppercase
		m_uniqueID = m_uniqueID.toUpperCase();

通过以上算法,可以产生32位的16进制数据,如:

49E9521A48A16EE6BCB3E290124BEE0A

需要说明,这个唯一码和手机架构平台有关,不同平台也会不同。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值