使用ExpandableListView / SimpleCursorTreeAdapter在UI线程上运行SQLite查询

时间:2011-01-12 17:57:00

标签: android cursor expandablelistview ui-thread

我正在开发用于显示大量RSS源的Android应用程序(是的,我知道有很多这样的应用程序已经存在)。要显示的数据由内容提供商支持,我希望向后兼容API级别4。

我正在使用ExpandableListView来显示三种不同RSS源的内容。 ExpandableListView的适配器实现为SimpleCursorTreeAdapter的子类:

private class RssFeedLatestListAdapter extends SimpleCursorTreeAdapter {

    public static final String FEED_NAME_COLUMN = "feedName";

    public RssFeedLatestListAdapter(Context ctx, Cursor groupCursor, int groupLayout,
            String[] groupFrom, int[] groupTo, int childLayout, String[] childFrom,
            int[] childTo) {
        super(ctx, groupCursor, groupLayout, groupLayout, groupFrom, groupTo, childLayout, childFrom, childTo);
    }

    @Override
    protected Cursor getChildrenCursor(final Cursor groupCursor) {
        final String feedName = groupCursor.getString(groupCursor.getColumnIndex(FEED_NAME_COLUMN));
        return managedQuery(LATEST_URI, null,
                FeedItemColumns.CATEGORY + " = ? AND " + FeedItemColumns.FEED + " = ?",
                new String[] { mCategory.getId(), feedName }, null);
    }

    @Override
    protected void bindGroupView(View view, Context context, Cursor cursor, boolean isExpanded) {
        super.bindGroupView(view, context, cursor, isExpanded);

        // Bind group view (impl details not important) ...
    }

    @Override
    protected void bindChildView(View view, Context context, Cursor cursor, boolean isLastChild) {
        super.bindChildView(view, context, cursor, isLastChild);

        // Bind child view (impl details not important) ...
    }
}

使用此设置,将按预期加载所有内容。但是,在加载列表内容期间, UI会随机挂起/断断续续。通常不足以获得ANR,但仍然明显且非常烦人!

为了调试这个问题,我启用了android.os.StrictMode(很棒的新功能/工具btw!),并启动了模拟器(在Android 2.3上)。加载列表内容后,我得到以下StrictMode输出:

D/StrictMode(  395): StrictMode policy violation; ~duration=1522 ms: android.os.StrictMode$StrictModeDiskReadViolation: policy=23 violation=2
D/StrictMode(  395):  at android.os.StrictMode$AndroidBlockGuardPolicy.onReadFromDisk(StrictMode.java:745)
D/StrictMode(  395):  at android.database.sqlite.SQLiteDatabase.rawQueryWithFactory(SQLiteDatabase.java:1345)
D/StrictMode(  395):  at android.database.sqlite.SQLiteQueryBuilder.query(SQLiteQueryBuilder.java:330)
D/StrictMode(  395):  at com.mycompany.myapp.provider.RSSFeedProvider.query(RSSFeedProvider.java:128)
D/StrictMode(  395):  at android.content.ContentProvider$Transport.query(ContentProvider.java:187)
D/StrictMode(  395):  at android.content.ContentResolver.query(ContentResolver.java:262)
D/StrictMode(  395):  at android.app.Activity.managedQuery(Activity.java:1550)
D/StrictMode(  395):  at com.mycompany.myapp.activity.MultipleFeedsActivity$RssFeedLatestListAdapter.getChildrenCursor(MultipleFeedsActivity.java:388)
D/StrictMode(  395):  at android.widget.CursorTreeAdapter.getChildrenCursorHelper(CursorTreeAdapter.java:106)
D/StrictMode(  395):  at android.widget.CursorTreeAdapter.getChildrenCount(CursorTreeAdapter.java:178)
D/StrictMode(  395):  at android.widget.ExpandableListConnector.refreshExpGroupMetadataList(ExpandableListConnector.java:561)
D/StrictMode(  395):  at android.widget.ExpandableListConnector.expandGroup(ExpandableListConnector.java:682)
D/StrictMode(  395):  at android.widget.ExpandableListConnector.expandGroup(ExpandableListConnector.java:636)
D/StrictMode(  395):  at android.widget.ExpandableListView.expandGroup(ExpandableListView.java:608)
D/StrictMode(  395):  at com.mycompany.myapp.activity.MultipleFeedsActivity.onReceiveResult(MultipleFeedsActivity.java:335)
D/StrictMode(  395):  at com.mycompany.myapp.service.FeedResultReceiver.onReceiveResult(FeedResultReceiver.java:40)
D/StrictMode(  395):  at android.os.ResultReceiver$MyRunnable.run(ResultReceiver.java:43)
D/StrictMode(  395):  at android.os.Handler.handleCallback(Handler.java:587)
D/StrictMode(  395):  at android.os.Handler.dispatchMessage(Handler.java:92)
D/StrictMode(  395):  at android.os.Looper.loop(Looper.java:123)
D/StrictMode(  395):  at android.app.ActivityThread.main(ActivityThread.java:3647)
D/StrictMode(  395):  at java.lang.reflect.Method.invokeNative(Native Method)
D/StrictMode(  395):  at java.lang.reflect.Method.invoke(Method.java:507)
D/StrictMode(  395):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
D/StrictMode(  395):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
D/StrictMode(  395):  at dalvik.system.NativeStart.main(Native Method)

这似乎合情合理!在SimpleCursorTreeAdapter.getChildrenCursor()中,在UI线程上查询内容提供者,这当然是一个坏主意,绝对会挂起UI!

我查看了CursorTreeAdapter({1}}的超级SimpleCursorTreeAdapter的{​​{3}}(API等级4),getChildrenCursor()的{​​{1}}说:

“如果要异步查询提供程序以防止阻止UI,可以返回null,稍后再调用setChildrenCursor(int,Cursor)。”

大!我们这样做!我更改了getChildrenCursor()的实现,以启动一个新任务,负责执行查询并在适配器上设置子游标。启动任务后,我只在getChildrenCursor()中返回null:

@Override
protected Cursor getChildrenCursor(final Cursor groupCursor) {
    final String feedName = groupCursor.getString(groupCursor.getColumnIndex(FEED_NAME_COLUMN));
    new RefreshChildrenCursorTask(groupCursor.getPosition()).execute(feedName);
    return null;
}

,其中RefreshChildrenCursorTask实现为:

private class RefreshChildrenCursorTask extends AsyncTask<String, Void, Cursor> {

    private int mGroupPosition;

    public RefreshChildrenCursorTask(int groupPosition) {
        this.mGroupPosition = groupPosition;
    }

    @Override
    protected Cursor doInBackground(String... params) {
        String feedName = params[0];
        return managedQuery(LATEST_URI, null,
                FeedItemColumns.CATEGORY + " = ? AND " + FeedItemColumns.FEED + " = ?",
                new String[] { mCategory.getId(), feedName }, null);
        }

        @Override
        protected void onPostExecute(Cursor childrenCursor) {
            mLatestListAdapter.setChildrenCursor(mGroupPosition, childrenCursor);
        }
    }
}

我在2.3模拟器上重新安装了该应用并启动了它。它就像一个魅力,再见无响应的用户界面!但欢乐并没有持续多久......接下来要做的就是在目标设备上运行相同的代码(在这种情况下,运行Android 2.2的三星Galaxy S)。然后,我在加载列表供稿内容时获得了以下Exception

E/AndroidRuntime(23191): FATAL EXCEPTION: main
E/AndroidRuntime(23191): java.lang.NullPointerException
E/AndroidRuntime(23191):  at android.widget.SimpleCursorTreeAdapter.initFromColumns(SimpleCursorTreeAdapter.java:194)
E/AndroidRuntime(23191):  at android.widget.SimpleCursorTreeAdapter.initChildrenFromColumns(SimpleCursorTreeAdapter.java:205)
E/AndroidRuntime(23191):  at android.widget.SimpleCursorTreeAdapter.init(SimpleCursorTreeAdapter.java:186)
E/AndroidRuntime(23191):  at android.widget.SimpleCursorTreeAdapter.(SimpleCursorTreeAdapter.java:136)
E/AndroidRuntime(23191):  at com.mycompany.myapp.activity.MultipleFeedsActivity$RssFeedLatestListAdapter.(MultipleFeedsActivity.java:378)
E/AndroidRuntime(23191):  at com.mycompany.myapp.activity.MultipleFeedsActivity$2.run(MultipleFeedsActivity.java:150)
E/AndroidRuntime(23191):  at android.os.Handler.handleCallback(Handler.java:587)
E/AndroidRuntime(23191):  at android.os.Handler.dispatchMessage(Handler.java:92)
E/AndroidRuntime(23191):  at android.os.Looper.loop(Looper.java:123)
E/AndroidRuntime(23191):  at android.app.ActivityThread.main(ActivityThread.java:4627)
E/AndroidRuntime(23191):  at java.lang.reflect.Method.invokeNative(Native Method)
E/AndroidRuntime(23191):  at java.lang.reflect.Method.invoke(Method.java:521)
E/AndroidRuntime(23191):  at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:871)
E/AndroidRuntime(23191):  at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:629)
E/AndroidRuntime(23191):  at dalvik.system.NativeStart.main(Native Method)

快速查看SimpleCursorTreeAdapter的源代码告诉我,它无法应对从getChildrenCursor()异步返回的null。他们似乎已经在Android 2.3中解决了这个问题,但是你应该如何在早期版本的Android中解决它?我是否真的必须编写自己的CursorTreeAdapter实现才能异步刷新子游标?

有没有人遇到同样的问题,甚至可能找到(可行的)解决方法?如果没有,有人可以给我一个关于如何继续的暗示!? 我真的很感激任何帮助我能得到!

提前致谢!

此致 雅各布

2 个答案:

答案 0 :(得分:2)

如果他们在2.3 SimpleCursorTreeAdapter中修复了它,为什么不直接下载java文件并将其包含在项目中并使用该版本而不是手机版本?

我没有查看代码,但只要他们没有进行任何新的2.3 API调用,就应该有效。

答案 1 :(得分:2)

感谢您的反馈,mp2526!我为我迟到的回应道歉!

这是一个很好的建议!但是,我真的认为应该可以在API 8中执行此操作(异步检索子游标),并且我误解了任何内容......显然,这是不可能的(至少不能使用{{ 1}})!

无论如何,我在{8}的API 8和API 9之间做了差异,他们在Android 2.3中所做的就是SimpleCursorTreeAdapterSimpleCursorTreeAdapter列id现在懒得初始化,即不再在构造函数中。相反,这些成员现在在第一次调用childFrom时初始化。这样,现在有时间异步检索子游标。由于API 4和API 9之间似乎没有对此类的公共API进行任何更改(新方法groupFrom除外),因此使用此类的API版本9的解决方案应该无需更改!我试了一下,确实做到了!

非常感谢,mp2526!