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

时间:2012-07-22 23:44:42

标签: java android multithreading user-interface android-contentresolver

我有一个应用需要从服务器提取数据并将其插入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中的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();
    }
}

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

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

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

4 个答案:

答案 0 :(得分:52)

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

答案 1 :(得分:3)

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

答案 2 :(得分:0)

Man.Relax yourself.And任何东西看起来都会更好。 首先,如果要启动新线程,则启动线程是Func start而不是Func run 不仅要调用func run。

new Thread(Runnable runnable).start();

然后我打赌使用Handler有时会比AsyncTask更好。

答案 3 :(得分:0)

您可以在AsynTaskdoInBackground(Integer int)重写方法中运行查询,并在onPostExecute(Integer int)方法上更新主界面。