ContentProvider insert() 总是在 UI 线程上运行?

Posted

技术标签:

【中文标题】ContentProvider insert() 总是在 UI 线程上运行?【英文标题】:ContentProvider insert() always runs on UI thread? 【发布时间】:2012-07-21 05:44:29 【问题描述】:

我有一个应用程序需要从服务器提取数据并将其插入 SQLite 数据库以响应用户输入。我认为这将非常简单——从服务器提取数据的代码是 AsyncTask 的一个相当简单的子类,它完全按照我的预期工作而无需挂起 UI 线程。我用一个简单的接口为它实现了回调功能,并将它包装在一个静态类中,所以我的代码如下所示:

MyServerCaller.getFolderContents(folderId, new OnFolderContentsResponseListener() 
    @Override
    public void onFolderContentsResponse(final List<FilesystemEntry> contents) 
        // do something with contents
    

一切都还好。即使服务器需要一个小时来检索数据,UI 仍然可以顺利运行,因为 getFolderContents 中的代码是在 AsyncTask 的 doInBackground 方法中运行的(它在与 UI 不同的线程中)。在 getFolderContents 方法的最后,调用 onFolderContentsResponse 并传递从服务器接收到的 FilesystemEntry 列表。我只是真的这么说,希望我的问题不在于 getFolderContents 方法或我的任何网络代码中,因为它不会在那里发生。

当我尝试通过 onFolderContentsResponse 方法中的 ContentProvider 子类插入数据库时​​出现问题; UI 总是在代码执行时挂起,这让我相信,尽管是从 AsyncTask 的 doInBackground 方法调用的,插入仍然以某种方式在 UI 线程上运行。有问题的代码如下所示:

MyServerCaller.getFolderContents(folderId, new OnFolderContentsResponseListener() 
    @Override
    public void onFolderContentsResponse(final List<FilesystemEntry> contents) 
        insertContentsIntoDB(contents);
    

还有insertContentsIntoDB 方法:

void insertContentsIntoDB(final List<FilesystemEntry> contents) 
    for (FilesystemEntry entry : contents) 
        ContentValues values = new ContentValues();
        values.put(COLUMN_1, entry.attr1);
        values.put(COLUMN_2, entry.attr2);
        // etc.

        mContentResolver.insert(MyContentProvider.CONTENT_URI, values);
    

其中 mContentResolver 先前已设置为 getContentResolver() 方法的结果。

我尝试将 insertContentsIntoDB 放在它自己的线程中,如下所示:

MyServerCaller.getFolderContents(folderId, new OnFolderContentsResponseListener() 
    @Override
    public void onFolderContentsResponse(final List<FilesystemEntry> contents) 
        new Thread(new Runnable() 
            @Override
            public void run() 
                insertContentsIntoDB(contents);
            
        ).run();
    

我也尝试在自己的线程中运行每个单独的插入(MyContentProvider 中的插入方法是同步的,所以这应该不会导致任何问题):

void insertContentsIntoDB(final List<FilesystemEntry> contents) 
    for (FilesystemEntry entry : contents) 
        new Thread(new Runnable() 
            @Override
            public void run() 
                ContentValues values = new ContentValues();
                values.put(COLUMN_1, entry.attr1);
                values.put(COLUMN_2, entry.attr2);
                // etc.
                mContentResolver.insert(MyContentProvider.CONTENT_URI, values);
            
        ).run();
    

为了更好地衡量,我还使用另一个 AsyncTask 的 doInBackground 方法中的相关代码尝试了这两种解决方案。最后,我在我的 androidManifest.xml 中明确地将 MyContentProvider 定义为存在于一个单独的进程中:

<provider android:name=".MyContentProvider" android:process=":remote"/>

它运行良好,但它似乎仍然在 UI 线程中运行。这就是我真正开始为此烦恼的地方,因为这对我来说根本没有任何意义。无论我做什么,UI 总是在插入过程中挂起。有什么办法让他们不这样做吗?

【问题讨论】:

从您的 ContentProvider 打印当前线程 ID/名称以验证您的假设。和/或使用调试器并检查正在运行的线程。 【参考方案1】:

不要调用mContentResolver.insert(),而是使用AsyncQueryHandler 及其startInsert() 方法。 AsyncQueryHandler 旨在促进异步 ContentResolver 查询。

【讨论】:

非常感谢,这正是我想要的! 由于某种原因,我的 AsyncQueryHandler 的 onQueryComplete() 方法永远不会被调用... call() 方法呢,是否也有一个异步类呢? 有时 onQueryComplete() 没有被调用!所以很烦人!【参考方案2】:

我认为您最初的问题可能是您在新线程上调用run 方法(这会导致在当前线程上继续执行)而不是调用start 方法。我认为这就是 Bright Great 在他/她的回答中试图表达的意思。见Difference between running and starting a thread。这是一个常见的错误。

【讨论】:

【参考方案3】:

Man.放松自己。任何事情都会看起来更好。 首先,Start a Thread 是 Func start 不是 Func run,如果要启动新的 Thread 不仅调用func运行。

new Thread(Runnable runnable).start();

那么我敢打赌,有时使用 Handler 会比 AsyncTask 更好。

【讨论】:

处理程序始终在其父线程中运行,而不是在后台运行。【参考方案4】:

您可以在AsynTask 的覆盖方法doInBackground(Integer int) 中运行查询,并在onPostExecute(Integer int) 方法上更新主UI。

【讨论】:

以上是关于ContentProvider insert() 总是在 UI 线程上运行?的主要内容,如果未能解决你的问题,请参考以下文章

Android ContentProvider 使用

Android ContentProvider 使用

ContentProvider原理分析

为啥我在 Xamarin 中使用 ContentProvider 插入项目后数据库为空?

简单的学习心得:网易云课堂Android开发第六章SQLite与ContentProvider

四大组件之ContentProvider-ContentProvider的数据存储