如何獲取Android唯一標識(唯一序列號)
有很多場景和需求你需要用到手機裝置的唯一識別符號。
在Android中,有以下幾種方法獲取這樣的ID。
1. The IMEI: 僅僅只對Android手機有效:
1 2 |
TelephonyManager TelephonyMgr = (TelephonyManager)getSystemService(TELEPHONY_SERVICE); String szImei = TelephonyMgr.getDeviceId(); |
採用此種方法,需要在AndroidManifest.xml中加入一個許可:android.permission.READ_PHONE_STATE,並且使用者應當允許安裝此應用。作為手機來講,IMEI是唯一的,它應該類似於 359881030314356(除非你有一個沒有量產的手機(水貨)它可能有無效的IMEI,如:0000000000000)。
2. Pseudo-Unique ID, 這個在任何Android手機中都有效
有一些特殊的情況,一些如平板電腦的設定沒有通話功能,或者你不願加入READ_PHONE_STATE許可。而你仍然想獲得唯一序列號之類的東西。這時你可以通過取出ROM版本、製造商、CPU型號、以及其他硬體資訊來實現這一點。這樣計算出來的ID不是唯一的(因為如果兩個手機應用了同樣的硬體以及Rom 映象)。但應當明白的是,出現類似情況的可能性基本可以忽略。要實現這一點,你可以使用Build類:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
String m_szDevIDShort |
大多數的Build成員都是字串形式的,我們只取他們的長度資訊。我們取到13個數字,並在前面加上“35”。這樣這個ID看起來就和15位IMEI一樣了。
3. The Android ID
通常被認為不可信,因為它有時為null。開發文件中說明了:這個ID會改變如果進行了出廠設定。並且,如果某個Andorid手機被Root過的話,這個ID也可以被任意改變。
1 |
String m_szAndroidID = Secure.getString(getContentResolver(), Secure.ANDROID_ID); |
Returns: 9774d56d682e549c . 無需任何許可。
4. The WLAN MAC Address string
是另一個唯一ID。但是你需要為你的工程加入android.permission.ACCESS_WIFI_STATE 許可權,否則這個地址會為null。
1 2 |
WifiManager wm = (WifiManager)getSystemService(Context.WIFI_SERVICE); String m_szWLANMAC = wm.getConnectionInfo().getMacAddress(); |
Returns: 00:11:22:33:44:55 (這不是一個真實的地址。而且這個地址能輕易地被偽造。).WLan不必開啟,就可讀取些值。
5. The BT MAC Address string
只在有藍芽的裝置上執行。並且要加入android.permission.BLUETOOTH 許可權.
1 2 3 |
BluetoothAdapter m_BluetoothAdapter = null; // Local Bluetooth adapter m_BluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); String m_szBTMAC = m_BluetoothAdapter.getAddress(); |
Returns: 43:25:78:50:93:38 . 藍芽沒有必要開啟,也能讀取。
Combined Device ID
綜上所述,我們一共有五種方式取得裝置的唯一標識。它們中的一些可能會返回null,或者由於硬體缺失、許可權問題等獲取失敗。
但你總能獲得至少一個能用。所以,最好的方法就是通過拼接,或者拼接後的計算出的MD5值來產生一個結果。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
String m_szLongID = m_szImei + m_szDevIDShort + m_szAndroidID+ m_szWLANMAC + m_szBTMAC; // compute md5 MessageDigest m = null; try { m = MessageDigest.getInstance("MD5"); } catch (NoSuchAlgorithmException e) { e.printStackTrace(); } m.update(m_szLongID.getBytes(),0,m_szLongID.length()); // get md5 bytes byte p_md5Data[] = m.digest(); // create a hex string String m_szUniqueID = 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_szUniqueID+="0"; // add number to string m_szUniqueID+=Integer.toHexString(b); } // hex string to uppercase m_szUniqueID = m_szUniqueID.toUpperCase(); |
通過以上演算法,可產生32位的16進位制資料:
9DDDF85AFF0A87974CE4541BD94D5F55
現在你就可以對其進行你的應用了。