Android?APN數(shù)據(jù)庫查詢對比分析(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)對Telephony.Carriers表的訪問增加了權(quán)限限制:
- 需要
Carrier Privileges或WRITE_APN_SETTINGS權(quán)限 - 查詢結(jié)果可能返回空Cursor,需配合
SubscriptionManager處理多SIM卡場景
分析
通常,Android的ContentProvider處理數(shù)據(jù)請求,而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)到主線程
線程行為對比表
| 調(diào)用方式 | 執(zhí)行線程 | 是否阻塞UI | 推薦場景 |
|---|---|---|---|
| 直接調(diào)用query() | 調(diào)用線程 | 可能阻塞 | 小型數(shù)據(jù)集/非UI線程調(diào)用 |
| CursorLoader自動(dòng)執(zhí)行query | AsyncTask線程池 | 無阻塞 | 列表數(shù)據(jù)加載等標(biāo)準(zhǔn)場景 |
對疑問的代碼執(zhí)行流程驗(yàn)證
原生CursorLoader實(shí)現(xiàn)原理(簡化版):
// Framework簡化的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)用線程的測試代碼
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ù)庫查詢在AsyncTask線程池執(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ù)庫查詢對比(APN案例)的文章就介紹到這了,更多相關(guān)Android APN數(shù)據(jù)庫查詢內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android實(shí)現(xiàn)用代碼簡單安裝和卸載APK的方法
這篇文章主要介紹了Android實(shí)現(xiàn)用代碼簡單安裝和卸載APK的方法,涉及Android針對APK文件及package的相關(guān)操作技巧,需要的朋友可以參考下2016-08-08
Android布局耗時(shí)監(jiān)測的三種實(shí)現(xiàn)方式
在Android應(yīng)用開發(fā)中,性能優(yōu)化是一個(gè)至關(guān)重要的方面,為了更好地監(jiān)測布局渲染的耗時(shí),我們需要一種可靠的實(shí)現(xiàn)方案,本文將介紹三種針對Android布局耗時(shí)監(jiān)測的實(shí)現(xiàn)方案,幫助開發(fā)者及時(shí)發(fā)現(xiàn)并解決布局性能問題,需要的朋友可以參考下2024-03-03
Android進(jìn)階教程之ViewGroup自定義布局
這篇文章主要給大家介紹了關(guān)于Android進(jìn)階教程之ViewGroup自定義布局的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對各位Android開發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06
Android RecycleView和線型布局制作聊天布局
大家好,本篇文章主要講的是Android RecycleView和線型布局制作聊天布局,感興趣的同學(xué)趕緊來看一看吧,對你有幫助的話記得收藏一下2022-01-01
Android的Launcher啟動(dòng)器中添加快捷方式及小部件實(shí)例
這篇文章主要介紹了在Android的Launcher啟動(dòng)器中添加快捷方式及窗口小部件的方法,包括在自己的應(yīng)用程序中添加窗口小部件AppWidget的例子,需要的朋友可以參考下2016-02-02
Flutter使用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)用的問題
本篇文章是對關(guān)于android創(chuàng)建快捷方式會(huì)啟動(dòng)兩個(gè)應(yīng)用的問題進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-06-06

