Android?APN數(shù)據(jù)庫查詢對(duì)比分析(APN案例)
功能背景
APN 數(shù)據(jù)通常存儲(chǔ)在數(shù)據(jù)庫中,由TelephonyProvider提供。當(dāng)用戶進(jìn)入APN設(shè)置界面時(shí),Activity會(huì)啟動(dòng),AOSP源碼通過ContentResolver查詢APN數(shù)據(jù)。關(guān)鍵分析點(diǎn)在于這個(gè)查詢操作是否在主線程執(zhí)行,因?yàn)橹骶€程上的耗時(shí)操作會(huì)導(dǎo)致ANR。
技術(shù)演進(jìn)補(bǔ)充
自Android 9(API 28)起,系統(tǒng)對(duì)Telephony.Carriers
表的訪問增加了權(quán)限限制:
- 需要
Carrier Privileges
或WRITE_APN_SETTINGS
權(quán)限 - 查詢結(jié)果可能返回空Cursor,需配合
SubscriptionManager
處理多SIM卡場(chǎng)景
分析
通常,Android的ContentProvider處理數(shù)據(jù)請(qǐng)求,而TelephonyProvider繼承自ContentProvider。查詢操作可能會(huì)在后臺(tái)線程執(zhí)行,但需要確認(rèn)。例如,CursorLoader會(huì)在后臺(tái)線程處理查詢,但如果是直接使用ContentResolver.query,可能就在主線程了。
Android源碼中
- 使用LoaderManager來異步加載數(shù)據(jù),這樣查詢會(huì)在后臺(tái)進(jìn)行。
- 如果用戶使用的是直接查詢而沒有異步處理,那么主線程可能被阻塞。
排查建議
檢查ApnSettingsActivity的onCreate或onResume方法,看是否觸發(fā)數(shù)據(jù)加載。
- 使用LoaderManager.initLoader來啟動(dòng)CursorLoader,這樣查詢會(huì)在后臺(tái)線程執(zhí)行。
- 否則,如果直接在主線程調(diào)用getContentResolver().query,就會(huì)在主線程執(zhí)行數(shù)據(jù)庫操作。
疑問
Q:getContentResolver().query都是在主線程執(zhí)行的嗎?
A:直接調(diào)用getContentResolver().query取決于調(diào)用線程。
根據(jù)Android文檔,ContentResolver的query方法默認(rèn)是在調(diào)用線程執(zhí)行的。如果直接在主線程調(diào)用getContentResolver().query,那么查詢會(huì)在主線程執(zhí)行,可能導(dǎo)致UI卡頓甚至ANR。然而,當(dāng)使用CursorLoader時(shí),內(nèi)部會(huì)使用AsyncTask或者類似的機(jī)制,在后臺(tái)線程執(zhí)行查詢,從而避免主線程阻塞。
CursorLoader內(nèi)部封裝了異步查詢,所以即使調(diào)用了ContentResolver的query方法,也是在Loader管理的后臺(tái)線程中執(zhí)行的,而不是主線程。
正確使用Loader或現(xiàn)代的替代方案(如Room的LiveData + ViewModel)來確保數(shù)據(jù)庫操作不在主線程進(jìn)行。
關(guān)鍵結(jié)論
- 直接調(diào)用:若在Activity/Fragment主線程中直接調(diào)用
getContentResolver().query()
,確實(shí)會(huì)在主線程執(zhí)行數(shù)據(jù)庫操作,可能導(dǎo)致ANR - 通過Loader機(jī)制:當(dāng)使用
CursorLoader
時(shí),系統(tǒng)自動(dòng)在后臺(tái)線程執(zhí)行查詢,通過Handler
將結(jié)果回調(diào)到主線程
線程行為對(duì)比表
調(diào)用方式 | 執(zhí)行線程 | 是否阻塞UI | 推薦場(chǎng)景 |
---|---|---|---|
直接調(diào)用query() | 調(diào)用線程 | 可能阻塞 | 小型數(shù)據(jù)集/非UI線程調(diào)用 |
CursorLoader自動(dòng)執(zhí)行query | AsyncTask線程池 | 無阻塞 | 列表數(shù)據(jù)加載等標(biāo)準(zhǔn)場(chǎng)景 |
對(duì)疑問的代碼執(zhí)行流程驗(yàn)證
原生CursorLoader實(shí)現(xiàn)原理(簡(jiǎn)化版):
// Framework簡(jiǎn)化的Loader線程切換邏輯 public class CursorLoader extends AsyncTaskLoader<Cursor> { @Override public Cursor loadInBackground() { // 在AsyncTask線程執(zhí)行 return getContext().getContentResolver().query(uri, ...); } @Override protected void onStartLoading() { forceLoad(); // 觸發(fā)異步加載 } }
主線程驗(yàn)證代碼:
// 驗(yàn)證調(diào)用線程的測(cè)試代碼 new Handler(Looper.getMainLooper()).post(() -> { // 在主線程執(zhí)行查詢 Cursor cursor = getContentResolver().query(Carriers.CONTENT_URI, ...); Log.d("ThreadTest", "MainThread query: " + (Looper.myLooper() == Looper.getMainLooper())); }); // 輸出結(jié)果:ThreadTest: MainThread query: true
代碼實(shí)現(xiàn)
優(yōu)化設(shè)想
用戶打開界面,Activity初始化Loader,LoaderManager啟動(dòng)CursorLoader,CursorLoader在后臺(tái)線程執(zhí)行查詢,通過ContentResolver調(diào)用TelephonyProvider的query方法,最終獲取APN數(shù)據(jù)并返回給主線程更新UI。

%% APN Settings界面數(shù)據(jù)加載時(shí)序圖 sequenceDiagram participant User participant ApnSettingsActivity participant LoaderManager participant CursorLoader participant TelephonyProvider participant Database User->>ApnSettingsActivity: 啟動(dòng)APN設(shè)置界面 activate ApnSettingsActivity ApnSettingsActivity->>LoaderManager: initLoader(APN_LOADER_ID) LoaderManager->>CursorLoader: 創(chuàng)建新Loader實(shí)例 activate CursorLoader CursorLoader->>TelephonyProvider: 異步執(zhí)行query() activate TelephonyProvider TelephonyProvider->>Database: 執(zhí)行SQL查詢 activate Database Database-->>TelephonyProvider: 返回APN數(shù)據(jù)Cursor deactivate Database TelephonyProvider-->>CursorLoader: 返回查詢結(jié)果 deactivate TelephonyProvider CursorLoader-->>LoaderManager: 交付結(jié)果 deactivate CursorLoader LoaderManager->>ApnSettingsActivity: onLoadFinished() ApnSettingsActivity->>ApnSettingsActivity: 更新UI列表 deactivate ApnSettingsActivity Note right of CursorLoader: 關(guān)鍵路徑說明<br/>1. CursorLoader自動(dòng)處理后臺(tái)線程<br/>2. 數(shù)據(jù)庫查詢?cè)贏syncTask線程池執(zhí)行<br/>3. 結(jié)果通過Handler返回主線程
如下是優(yōu)化方案的案例,但是原生邏輯并不是直接一個(gè)Activity
package com.android.settings.network.apn; // APN數(shù)據(jù)庫查詢不會(huì)阻塞主線程,通過CursorLoader機(jī)制實(shí)現(xiàn) // 實(shí)際查詢發(fā)生在AsyncTask線程(AsyncTask.THREAD_POOL_EXECUTOR) // 結(jié)果回調(diào)通過Handler機(jī)制返回主線程 // ApnSettings.java 核心邏輯 public class ApnSettings extends PreferenceActivity implements LoaderManager.LoaderCallbacks<Cursor> { @Override protected void onCreate(Bundle savedInstanceState) { getLoaderManager().initLoader(APN_LOADER_ID, null, this); // 啟動(dòng)異步加載 } @Override public Loader<Cursor> onCreateLoader(int id, Bundle args) { return new CursorLoader(this, Telephony.Carriers.CONTENT_URI, PROJECTION, null, null, Telephony.Carriers.DEFAULT_SORT_ORDER); } @Override public void onLoadFinished(Loader<Cursor> loader, Cursor data) { mAdapter.swapCursor(data); // 主線程更新UI } }
以上符合Android的最佳實(shí)踐,即避免在主線程進(jìn)行IO操作。
- ApnSettingsActivity使用了LoaderManager來初始化CursorLoader。
- 在onCreateLoader方法中創(chuàng)建了CursorLoader實(shí)例,參數(shù)包括ContentProvider的URI和查詢參數(shù)。
- 當(dāng)LoaderManager啟動(dòng)加載時(shí),CursorLoader會(huì)在后臺(tái)線程執(zhí)行查詢,完成后再通過onLoadFinished回調(diào)主線程更新UI。
AOSP
packages/apps/Settings/src/com/android/settings/network/apn/ApnSettings.java
/** Handle each different apn setting. */ public class ApnSettings extends RestrictedSettingsFragment implements Preference.OnPreferenceChangeListener { static final String TAG = "ApnSettings";
到此這篇關(guān)于Android 數(shù)據(jù)庫查詢對(duì)比(APN案例)的文章就介紹到這了,更多相關(guān)Android APN數(shù)據(jù)庫查詢內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android實(shí)現(xiàn)用代碼簡(jiǎn)單安裝和卸載APK的方法
這篇文章主要介紹了Android實(shí)現(xiàn)用代碼簡(jiǎn)單安裝和卸載APK的方法,涉及Android針對(duì)APK文件及package的相關(guān)操作技巧,需要的朋友可以參考下2016-08-08Android布局耗時(shí)監(jiān)測(cè)的三種實(shí)現(xiàn)方式
在Android應(yīng)用開發(fā)中,性能優(yōu)化是一個(gè)至關(guān)重要的方面,為了更好地監(jiān)測(cè)布局渲染的耗時(shí),我們需要一種可靠的實(shí)現(xiàn)方案,本文將介紹三種針對(duì)Android布局耗時(shí)監(jiān)測(cè)的實(shí)現(xiàn)方案,幫助開發(fā)者及時(shí)發(fā)現(xiàn)并解決布局性能問題,需要的朋友可以參考下2024-03-03Android進(jìn)階教程之ViewGroup自定義布局
這篇文章主要給大家介紹了關(guān)于Android進(jìn)階教程之ViewGroup自定義布局的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對(duì)各位Android開發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06Android RecycleView和線型布局制作聊天布局
大家好,本篇文章主要講的是Android RecycleView和線型布局制作聊天布局,感興趣的同學(xué)趕緊來看一看吧,對(duì)你有幫助的話記得收藏一下2022-01-01Android的Launcher啟動(dòng)器中添加快捷方式及小部件實(shí)例
這篇文章主要介紹了在Android的Launcher啟動(dòng)器中添加快捷方式及窗口小部件的方法,包括在自己的應(yīng)用程序中添加窗口小部件AppWidget的例子,需要的朋友可以參考下2016-02-02Flutter使用Overlay與ColorFiltered新手引導(dǎo)實(shí)現(xiàn)示例
這篇文章主要介紹了Flutter使用Overlay與ColorFiltered新手引導(dǎo)實(shí)現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10解析android創(chuàng)建快捷方式會(huì)啟動(dòng)兩個(gè)應(yīng)用的問題
本篇文章是對(duì)關(guān)于android創(chuàng)建快捷方式會(huì)啟動(dòng)兩個(gè)應(yīng)用的問題進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06