jcs*_*nt0 35 java user-interface multithreading android android-contentresolver
我有一个应用程序需要从服务器提取数据并将其插入SQLite数据库以响应用户输入.我认为这很简单 - 从服务器提取数据的代码是AsyncTask的一个相当简单的子类,它完全按照我的预期工作,而不会挂起UI线程.我使用一个简单的接口为它实现了回调功能,并将其包装在一个静态类中,所以我的代码如下所示:
MyServerCaller.getFolderContents(folderId, new OnFolderContentsResponseListener() {
@Override
public void onFolderContentsResponse(final List<FilesystemEntry> contents) {
// do something with contents
}
}
Run Code Online (Sandbox Code Playgroud)
一切都还不错.即使服务器需要一个小时来检索数据,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);
}
}
Run Code Online (Sandbox Code Playgroud)
和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);
}
}
Run Code Online (Sandbox Code Playgroud)
其中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();
}
}
Run Code Online (Sandbox Code Playgroud)
我也尝试在自己的线程中运行每个插入(MyContentProvider中的insert方法是同步的,所以这不应该导致任何问题):
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();
}
}
Run Code Online (Sandbox Code Playgroud)
只是为了好的衡量,我还尝试了这两个解决方案与另一个AsyncTask的doInBackground方法中的相关代码.最后,我明确地将MyContentProvider定义为生活在AndroidManifest.xml中的单独进程中:
<provider android:name=".MyContentProvider" android:process=":remote"/>
Run Code Online (Sandbox Code Playgroud)
它运行正常,但它似乎仍然在UI线程中运行.这就是我真正开始用头发撕掉头发的地方,因为这对我来说根本没有任何意义.无论我做什么,UI都会在插入过程中挂起.有没有办法让他们不去?
Gun*_*son 52
而不是调用mContentResolver.insert()
,使用AsyncQueryHandler
和它的startInsert()
方法.AsyncQueryHandler
旨在促进异步ContentResolver
查询.
归档时间: |
|
查看次数: |
9004 次 |
最近记录: |