Android-關於設備唯一ID的奇技淫巧
小知識,大挑戰!本文正在參與「程序員必備小知識」創作活動
本文已參與 「掘力星計劃」 ,贏取創作大禮包,挑戰創作激勵金。
前言
最近在二開項目國際版客户的功能,我們項目中默認是有一個遊客登錄的,一般大家都是取Android設備的唯一ID上傳服務器,然後服務器給你分配一個用户信息.但是Google在高版本對於設備唯一Id的獲取簡直限制到了極點.
以前我都是直接獲取IMEI來作為設備的唯一標識
var imei: String = ""
val tm: TelephonyManager =
context.getSystemService(Service.TELEPHONY_SERVICE) as TelephonyManager
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
imei = tm.imei
} else {
imei = tm.deviceId
}
Log.e("TAG","$imei")
imei和deviceId都有一個重載函數,主要是區別雙卡的一個情況
Android6.0以後我們加一個動態權限即可,但是用户只要拒絕就沒辦法獲取了,不過一般來説我們會有個彈框來引導用户同意
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
Android 10.0 谷歌再一次收緊權限
<uses-permission android:name="android.permission.READ_PRIVILEGED_PHONE_STATE" />
如果你把他放到AndroidManifest會報錯
官方也説了,你要是弟弟(9.0 以下)我給你報null,你要是10.0 還敢用我就直接拋異常. 後面在stackoverflow上面找到了一個辦法
``` public class DeviceUuidFactory { protected static final String PREFS_FILE = "device_id.xml"; protected static final String PREFS_DEVICE_ID = "device_id"; protected static UUID uuid;
public DeviceUuidFactory(Context context) {
if( uuid ==null ) {
synchronized (DeviceUuidFactory.class) {
if( uuid == null) {
final SharedPreferences prefs = context.getSharedPreferences( PREFS_FILE, 0);
final String id = prefs.getString(PREFS_DEVICE_ID, null );
if (id != null) {
// Use the ids previously computed and stored in the prefs file
uuid = UUID.fromString(id);
} else {
final String androidId = Secure.getString(context.getContentResolver(), Secure.ANDROID_ID);
// Use the Android ID unless it's broken, in which case fallback on deviceId,
// unless it's not available, then fallback on a random number which we store
// to a prefs file
try {
if () {
uuid = UUID.nameUUIDFromBytes(androidId.getBytes("utf8"));
} else {
@SuppressLint("MissingPermission") final String deviceId = ((TelephonyManager) context.getSystemService( Context.TELEPHONY_SERVICE )).getDeviceId();
uuid = deviceId!=null ? UUID.nameUUIDFromBytes(deviceId.getBytes("utf8")) : UUID.randomUUID();
}
} catch (UnsupportedEncodingException e) {
throw new RuntimeException(e);
}
// Write the value out to the prefs file
prefs.edit().putString(PREFS_DEVICE_ID, uuid.toString() ).commit();
}
}
}
}
}
/**
* Returns a unique UUID for the current android device. As with all UUIDs, this unique ID is "very highly likely"
* to be unique across all Android devices. Much more so than ANDROID_ID is.
*
* The UUID is generated by using ANDROID_ID as the base key if appropriate, falling back on
* TelephonyManager.getDeviceID() if ANDROID_ID is known to be incorrect, and finally falling back
* on a random UUID that's persisted to SharedPreferences if getDeviceID() does not return a
* usable value.
*
* In some rare circumstances, this ID may change. In particular, if the device is factory reset a new device ID
* may be generated. In addition, if a user upgrades their phone from certain buggy implementations of Android 2.2
* to a newer, non-buggy version of Android, the device ID may change. Or, if a user uninstalls your app on
* a device that has neither a proper Android ID nor a Device ID, this ID may change on reinstallation.
*
* Note that if the code falls back on using TelephonyManager.getDeviceId(), the resulting ID will NOT
* change after a factory reset. Something to be aware of.
*
* Works around a bug in Android 2.2 for many devices when using ANDROID_ID directly.
*
*
* @return a UUID that may be used to uniquely identify your device for most purposes.
*/
public String getDeviceUuid() {
return uuid.toString();
}
} ```
這個類的意思是,首先他會去SharedPreferences查詢有沒有,沒有的話再去查詢ANDROID_ID,後面判斷了是否是9774d56d682e549c,因為有的廠商手機好多ANDROID_ID都是這個,所以判斷一下,防止好幾萬個人用一個賬號,不然那就笑嘻嘻了,後面如果真等於9774d56d682e549c了,就通過下面的
@SuppressLint("MissingPermission") final String deviceId = ((TelephonyManager) context.getSystemService( Context.TELEPHONY_SERVICE )).getDeviceId();
來獲取DeviceId,但是這個AndroidId雖然可以是獲取了,但是會受限於簽名文件,如果在相同設備上運行但是應用簽名不一樣,獲取到的ANDROID_ID就會不一樣,比如谷歌商店會二次簽名apk,他獲取的id可能就是159951,後面我們要測試時,上傳到內部測試的包好像會再次簽名,這次獲取的可能是951159,然後我們用android提供的簽名文件可能就是147258,我們自己新建一個簽名文件就可能是258369,總之這個ANDROID_ID會受制於簽名文件
反正最後我們國際版用到了Mob的推送服務,推送中有一個只推送單個設備,然後我們就設想,直接用Mob的唯一設備Id和我們服務器綁定如何,後面一經測試,效果很好,直接跳過大堆測試和尋找時間
``` //阿里雲唯一設備id val deviceId = PushServiceFactory.getCloudPushService().deviceId
//Mob CloudPushService pushService = PushServiceFactory.getCloudPushService(); pushService.register(applicationContext, new CommonCallback() { @Override public void onSuccess(String response) { Log.e("TAG", "onSuccess: "+response); }
@Override
public void onFailed(String errorCode, String errorMessage) {
}
});
//友盟唯一設備ID val pushAgent = PushAgent.getInstance(context) pushAgent.register(object : UPushRegisterCallback { override fun onSuccess(deviceToken: String) { //註冊成功會返回deviceToken deviceToken是推送消息的唯一標誌 Log.i(TAG, "註冊成功:deviceToken:--> $deviceToken") }
override fun onFailure(errCode: String, errDesc: String) {
Log.e(TAG, "註冊失敗:--> code:$errCode, desc:$errDesc")
}
}) ```
這是常用的第三方服務獲取唯一設備ID的方法,其實有的人可能用的跟我不一樣,基本上文檔裏面都有,真找不到可以去問問客服
終於解決一個讓人頭疼的問題了,下班,回家
- 學習Android的第十七天
- 這是一個吸貓文章的標題,甚至可以有兩行這麼多哦
- 學習Android的第四天
- 學習Android的第一天
- 含有邊框的TextView-Android
- 電池-Android
- 倒計時封裝-Android
- 標題和狀態欄滑動漸變(2)-Android
- 標題和狀態欄滑動漸變(1)-Android
- 關於選項卡三方庫FlycoTabLayout的使用及修改
- 關於第三方庫SmartTabLayout的一點小修改
- 關於遞歸反轉鏈表的思路
- 搜索歷史記錄的實現-Android
- 感覺讓人耳目一新的動畫庫Lottie
- Android-關於設備唯一ID的奇技淫巧
- 稍微巧妙的雙模塊聯動-ViewPager
- 動畫庫NineOldAndroids實戰自定義懸浮窗
- 關於項目中圓角及特殊圓角的實際使用問題
- 自定義TextView可控制Drawable大小