Android 数据库查询对比(APN案例)

功能背景

APN 数据通常存储在数据库中,由TelephonyProvider提供。当用户进入APN设置界面时,Activity会启动,AOSP源码通过ContentResolver查询APN数据。关键分析点在于这个查询操作是否在主线程执行,因为主线程上的耗时操作会导致ANR。

技术演进补充

自Android 9(API 28)起,系统对Telephony.Carriers表的访问增加了权限限制:

  • 需要Carrier PrivilegesWRITE_APN_SETTINGS权限
  • 查询结果可能返回空Cursor,需配合SubscriptionManager处理多SIM卡场景

分析

通常,Android的ContentProvider处理数据请求,而TelephonyProvider继承自ContentProvider。查询操作可能会在后台线程执行,但需要确认。例如,CursorLoader会在后台线程处理查询,但如果是直接使用ContentResolver.query,可能就在主线程了。

Android源码中

  • 使用LoaderManager来异步加载数据,这样查询会在后台进行。
  • 如果用户使用的是直接查询而没有异步处理,那么主线程可能被阻塞。

排查建议

检查ApnSettingsActivity的onCreate或onResume方法,看是否触发数据加载。

  • 使用LoaderManager.initLoader来启动CursorLoader,这样查询会在后台线程执行。
  • 否则,如果直接在主线程调用getContentResolver().query,就会在主线程执行数据库操作。

疑问

Q:getContentResolver().query都是在主线程执行的吗?

A:直接调用getContentResolver().query取决于调用线程。

根据Android文档,ContentResolver的query方法默认是在调用线程执行的。如果直接在主线程调用getContentResolver().query,那么查询会在主线程执行,可能导致UI卡顿甚至ANR。然而,当使用CursorLoader时,内部会使用AsyncTask或者类似的机制,在后台线程执行查询,从而避免主线程阻塞。

CursorLoader内部封装了异步查询,所以即使调用了ContentResolver的query方法,也是在Loader管理的后台线程中执行的,而不是主线程。

正确使用Loader或现代的替代方案(如Room的LiveData + ViewModel)来确保数据库操作不在主线程进行。

关键结论

  1. 直接调用 :若在Activity/Fragment主线程中直接调用getContentResolver().query()确实会在主线程执行数据库操作,可能导致ANR
  2. 通过Loader机制 :当使用CursorLoader时,系统自动在后台线程执行查询,通过Handler将结果回调到主线程
调用方式 执行线程 是否阻塞UI 推荐场景
直接调用query() 调用线程 可能阻塞 小型数据集/非UI线程调用
CursorLoader自动执行query AsyncTask线程池 无阻塞 列表数据加载等标准场景
[线程行为对比表]

对疑问的代码执行流程验证

原生CursorLoader实现原理(简化版):

java 复制代码
// Framework简化的Loader线程切换逻辑
public class CursorLoader extends AsyncTaskLoader<Cursor> {
    @Override
    public Cursor loadInBackground() { // 在AsyncTask线程执行
        return getContext().getContentResolver().query(uri, ...);
    }
    
    @Override
    protected void onStartLoading() {
        forceLoad(); // 触发异步加载
    }
}

主线程验证代码:

java 复制代码
// 验证调用线程的测试代码
new Handler(Looper.getMainLooper()).post(() -> {
    // 在主线程执行查询
    Cursor cursor = getContentResolver().query(Carriers.CONTENT_URI, ...);
    Log.d("ThreadTest", "MainThread query: " + (Looper.myLooper() == Looper.getMainLooper()));
});

// 输出结果:ThreadTest: MainThread query: true

代码实现

优化设想

用户打开界面,Activity初始化Loader,LoaderManager启动CursorLoader,CursorLoader在后台线程执行查询,通过ContentResolver调用TelephonyProvider的query方法,最终获取APN数据并返回给主线程更新UI。

APN Settings界面数据优化加载时序图

复制代码
%% APN Settings界面数据加载时序图
sequenceDiagram
    participant User
    participant ApnSettingsActivity
    participant LoaderManager
    participant CursorLoader
    participant TelephonyProvider
    participant Database

    User->>ApnSettingsActivity: 启动APN设置界面
    activate ApnSettingsActivity
    ApnSettingsActivity->>LoaderManager: initLoader(APN_LOADER_ID)
    LoaderManager->>CursorLoader: 创建新Loader实例
    activate CursorLoader

    CursorLoader->>TelephonyProvider: 异步执行query()
    activate TelephonyProvider
    TelephonyProvider->>Database: 执行SQL查询
    activate Database
    Database-->>TelephonyProvider: 返回APN数据Cursor
    deactivate Database
    TelephonyProvider-->>CursorLoader: 返回查询结果
    deactivate TelephonyProvider

    CursorLoader-->>LoaderManager: 交付结果
    deactivate CursorLoader
    LoaderManager->>ApnSettingsActivity: onLoadFinished()
    ApnSettingsActivity->>ApnSettingsActivity: 更新UI列表
    deactivate ApnSettingsActivity

    Note right of CursorLoader: 关键路径说明<br/>1. CursorLoader自动处理后台线程<br/>2. 数据库查询在AsyncTask线程池执行<br/>3. 结果通过Handler返回主线程

如下是优化方案的案例,但是原生逻辑并不是直接一个Activity

java 复制代码
package com.android.settings.network.apn;


// APN数据库查询不会阻塞主线程,通过CursorLoader机制实现
// 实际查询发生在AsyncTask线程(AsyncTask.THREAD_POOL_EXECUTOR)
// 结果回调通过Handler机制返回主线程

// ApnSettings.java 核心逻辑
public class ApnSettings extends PreferenceActivity implements LoaderManager.LoaderCallbacks<Cursor> {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        getLoaderManager().initLoader(APN_LOADER_ID, null, this); // 启动异步加载
    }

    @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的最佳实践,即避免在主线程进行IO操作。

  • ApnSettingsActivity使用了LoaderManager来初始化CursorLoader。
  • 在onCreateLoader方法中创建了CursorLoader实例,参数包括ContentProvider的URI和查询参数。
  • 当LoaderManager启动加载时,CursorLoader会在后台线程执行查询,完成后再通过onLoadFinished回调主线程更新UI。

AOSP

packages/apps/Settings/src/com/android/settings/network/apn/ApnSettings.java

java 复制代码
/** Handle each different apn setting. */
public class ApnSettings extends RestrictedSettingsFragment
        implements Preference.OnPreferenceChangeListener {
    static final String TAG = "ApnSettings";
相关推荐
ii_best4 小时前
按键精灵支持安卓14、15系统,兼容64位环境开发辅助工具
android
美狐美颜sdk5 小时前
跨平台直播美颜SDK集成实录:Android/iOS如何适配贴纸功能
android·人工智能·ios·架构·音视频·美颜sdk·第三方美颜sdk
晋阳十二夜5 小时前
【压力测试之_Jmeter链接Oracle数据库链接】
数据库·oracle·压力测试
GDAL6 小时前
Node.js v22.5+ 官方 SQLite 模块全解析:从入门到实战
数据库·sqlite·node.js
DCTANT7 小时前
【原创】国产化适配-全量迁移MySQL数据到OpenGauss数据库
java·数据库·spring boot·mysql·opengauss
恋猫de小郭9 小时前
Meta 宣布加入 Kotlin 基金会,将为 Kotlin 和 Android 生态提供全新支持
android·开发语言·ios·kotlin
AI、少年郎10 小时前
Oracle 进阶语法实战:从多维分析到数据清洗的深度应用(第四课)
数据库·oracle
赤橙红的黄10 小时前
自定义线程池-实现任务0丢失的处理策略
数据库·spring
aqi0010 小时前
FFmpeg开发笔记(七十七)Android的开源音视频剪辑框架RxFFmpeg
android·ffmpeg·音视频·流媒体
DataGear10 小时前
如何在DataGear 5.4.1 中快速制作SQL服务端分页的数据表格看板
javascript·数据库·sql·信息可视化·数据分析·echarts·数据可视化