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

我有一个应用程序,需要从服务器中提取数据并将其插入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 放在它自己的 Thread 中,如下所示:

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方法中使用相关代码尝试了这两种解决方案。最后,我明确地将MyContentProvider定义为在我的AndroidManifest中生活在一个单独的进程中.xml:

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

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


答案 1

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


答案 2

我认为您最初的问题可能是您在新线程上调用该方法(这会导致在当前线程上继续执行),而不是调用该方法。我想这就是Bright Great在他/她的答案中想要表达的。请参阅运行线程和启动线程之间的区别。这是一个常见的错误。runstart


推荐