gpt4 book ai didi

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

转载 作者:IT老高 更新时间:2023-10-28 20:53:36 25 4
gpt4 key购买 nike

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

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 总是在插入过程中挂起。有什么办法让他们不这样做吗?

最佳答案

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

关于java - ContentProvider insert() 总是在 UI 线程上运行?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11604914/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com