1. 程式人生 > >Android 的裝置ID

Android 的裝置ID

Android 6.0(API Level 23)開始,應用程式的許可權有新的機制,在API 23之前(Android 4.x、5.x)應用程式在安裝時會要求所有會使用到的許可權,如INTERNET網路連線、READ_CONTACT讀取聯絡人資料或是WRITE_EXTERNAL_STORAGE寫入外部儲存裝置(如SD Card)等,在API 23開始,許可權分為兩大型別,一般許可權(Normal Permission)與危險許可權(Dangerous Permission)。

當應用程式在開發時,其「targetSdkVersion」目標開發版本設定為23或以上時,且在Android 6.0以上的手機環境執行時,對於許可權相關的運作方式說明如下。

危險許可權依照功能分為以下幾個組別:

序號 群組 專案 說明
1 CALENDAR日曆 READ_CALENDAR 讀取日曆
WRITE_CALENDAR 寫入日曆
2 CAMERA CAMERA 相機拍照功能
3 CONTACTS聯絡人 READ_CONTACTS 讀取聯絡人
WRITE_CONTACTS 寫入聯絡人
GET_ACCOUNTS 取得手機帳號
4 LOCATION位置 ACCESS_FINE_LOCATION 取得精確位置
ACCESS_COARSE_LOCATION 取得大約位置
5 MICROPHONE麥克風 RECORD_AUDIO 錄製聲音
6 PHONE電話 READ_PHONE_STATE 讀取通話狀態
CALL_PHONE 撥出電話
READ_CALL_LOG 讀取通話記錄
WRITE_CALL_LOG 寫入通話記錄
ADD_VOICEMAIL 新增語音留言
USE_SIP 使用SIP網路電話
PROCESS_OUTGOING_CALLS 存取撥出電話
7 SENSORS感應器 BODY_SENSORS 讀取體感資料
8 SMS簡訊 SEND_SMS 傳送簡訊
RECEIVE_SMS 接收簡訊
READ_SMS 讀取簡訊
RECEIVE_WAP_PUSH 接收WAP推播訊息
RECEIVE_MMS 接收多媒體簡訊
9 STORAGE儲存 READ_EXTERNAL_STORAGE 讀取外部儲存
WRITE_EXTERNAL_STORAGE 寫入外部儲存

在撰寫 Android App 時,時常需要呼叫一些平臺上的資訊,
例如使用 TelephonyManager 查詢 IEMI CODE 等需求,
就以取得 IMEI 為例,在 Android 中是使用 getDeviceId() 這個 API


public String getIMEI()
{
final TelephonyManager telephonyManager = getTelephonyManager();
if(telephonyManager == null)
{
return null;
}
final String imeiCode = telephonyManager.getDeviceId();
return (imeiCode!=null)? imeiCode : "Android_Emulator";
}

private TelephonyManager getTelephonyManager()
{
if(telephonyManager == null)
{
telephonyManager = (TelephonyManager)_service.getSystemService(Context.TELEPHONY_SERVICE);
}
return telephonyManager;
}

在一個空的 Android 專案中若這段 code 放在 try catch 中,
會發生跳到 catch block 的狀況,TRACE 後會發現是在 getDeviceID() 時發生錯誤,
這是因為部份 class 中的部份 method 在使用時必須擁有相關的許可權,
試試在 Android Developers 中找到 TelephyManager,再找到 getDeviceId()

仔細看過後發現底下的確寫著:Requires Permission: READ_PHONE_STATE

由於安卓平臺的開放性和碎片化,對於我們獲取唯一可靠的裝置識別符號帶來了很大的困難,在本片文章中我們將介紹安卓平臺下的,各種獲取裝置識別符號的方式以及這些方式的利弊,那什麼是裝置識別符號呢?
簡單說,裝置識別符號就是唯一標識該裝置的一串程式碼或符號。

裝置識別符號在Android開發中的作用

  • 跟蹤使用者安裝
  • 識別物理裝置

獲取Android裝置ID(DeviceId)

裝置ID(DeviceId)是手機唯一識別符號,屬於比較穩定的裝置識別符號。對於GSM返回IMEI,對於CDMA返回MEID或ESN。

//uses-permission android:name="android.permission.READ_PHONE_STATE"
private String getDeviceId() {
    TelephonyManager telephonyManager = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
    String deviceId = telephonyManager.getDeviceId();
    return deviceId;
}

缺陷:

  • 非手機裝置沒有這種唯一的識別符號,比如平板電腦、機頂盒、音樂播放器等。
  • 許可權問題:呼叫方法需要READ_PHONE_STATE許可權,特別是Android 6.0以後,許可權需要使用者手動確認。
  • Bug:在一些裝置製造商的裝置上,這個只是不準確的 ,比如會返回一堆0或者星號。

獲取Mac Address

通過檢索裝置的Wi-Fi或者藍芽硬體是有可能取得裝置的Mac地址的。

// uses-permission android:name="android.permission.ACCESS_WIFI_STATE"
private String getWifiMacDId() {
    WifiManager wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);
    String wifMacId = wifiManager.getConnectionInfo().getMacAddress();
    return wifMacId;
}

//uses-permission android:name="android.permission.BLUETOOTH" 
private String getBlueToothMacDId() {
    BluetoothAdapter bluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
    String blueToothMacId = bluetoothAdapter.getAddress();
    return blueToothMacId;
}

缺陷:

  • 硬體限制: 並不是所有的裝置都有WiFi和藍芽硬體
  • 許可權問題:需要ACCESS_WIFI_STATE或者BLUETOOTH許可權
  • 系統限制:為了保護使用者隱私,從Android 6.0開始將不能獲取WIFI和藍芽有效地Mac地址(02:0:00:00:00:0)

SERIAL

從Android 2.3 開始,可以獲取裝置的硬體序列號

private String getSerialId(){
      return android.os.Build.SERIAL;
}

缺陷:
Bug:可能不存在序列號的情況

Sim Serial Number 獲取SIM卡的序列號

//uses-permission android:name="android.permission.READ_PHONE_STATE"
private String getSimSerialNumber() {
    TelephonyManager telephonyManager = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
    String simSerialNumber = telephonyManager.getSimSerialNumber();
    return simSerialNumber;
}

缺陷:

  • 裝置限制:僅裝有SIM卡的裝置才能獲取到
  • 許可權問題:呼叫方法需要READ_PHONE_STATE許可權,特別是Android 6.0以後,許可權需要使用者手動確認。
  • Bug :對於CDMA裝置,返回的是一空值

Android ID

裝置第一次啟動時產生和儲存的 64-bit 字串

private String getAndroidId() {
    return Settings.Secure.getString(getContentResolver(), Settings.Secure.ANDROID_ID);
}

缺陷:

  • 恢復出廠設定後可能會重置該值。
  • 部分裝置由於製造商錯誤實現,導致返回相同的 Android_ID 。
  • 在 Android 4.2 及以上,裝置啟用多使用者功能後,每個使用者的 Android ID 不同。

UUID

全域性唯一識別符號,是指在一臺機器上生成的數字,它保證對在同一個時空中所有的機器都是唯一的

private String getUUID() {
    return java.util.UUID.randomUUID().toString();
}

缺陷:

  • 和裝置無關,不同的應用會產生不同的ID,無法做到裝置唯一

如何選擇裝置識別符號

  • 跟蹤使用者安裝,不需要嚴格意義上裝置唯一的ID,使用UUID即可跟蹤使用者的安裝
  • 識別物理裝置,使用多個硬體識別符號進行拼裝
private String getUUID() {
    return md5(androidId()+serialNumber()+" ");
}

識別模擬器

public static boolean runnigOnEmulator() {
    if (!TextUtils.isEmpty(Build.MODEL) && Build.MODEL.toLowerCase().contains("sdk")) {
        return true;
    }
    if (!TextUtils.isEmpty(Build.MANUFACTURER) && Build.MANUFACTURER.toLowerCase().contains("unknown")) {
        return true;
    }
    if (!TextUtils.isEmpty(Build.DEVICE) && Build.DEVICE.toLowerCase().contains("generic")) {
        return true;
    }
    return false;
}