2

我的应用程序在 Nexus 7 平板电脑上运行时遇到一个奇怪的问题。我有两个片段使用 CursorLoader 从我的自定义 ContentProvider 加载数据。

当我启动应用程序时,它在横向和纵向模式下都运行良好,加载程序正确加载数据。当我旋转屏幕时会出现问题:我的 Activity 和两个 Fragment 都已正确重新创建,并且加载程序再次启动,如预期的那样。他们加载数据(调试时我看到两个加载器都调用了 onLoadFinished),但之后应用程序是 FC,但有以下异常:

08-30 13:10:03.310: E/AndroidRuntime(14964): FATAL EXCEPTION: ModernAsyncTask #4
08-30 13:10:03.310: E/AndroidRuntime(14964): java.lang.RuntimeException: An error occured while executing doInBackground()
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.support.v4.content.ModernAsyncTask$3.done(Unknown Source)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at java.util.concurrent.FutureTask$Sync.innerSetException(FutureTask.java:273)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at java.util.concurrent.FutureTask.setException(FutureTask.java:124)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:307)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at java.util.concurrent.FutureTask.run(FutureTask.java:137)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at java.lang.Thread.run(Thread.java:856)
08-30 13:10:03.310: E/AndroidRuntime(14964): Caused by: java.lang.NullPointerException
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.content.ContentResolver.acquireUnstableProvider(ContentResolver.java:1094)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.content.ContentResolver.query(ContentResolver.java:354)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.content.ContentResolver.query(ContentResolver.java:313)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.support.v4.content.CursorLoader.loadInBackground(Unknown Source)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.support.v4.content.CursorLoader.loadInBackground(Unknown Source)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.support.v4.content.AsyncTaskLoader.onLoadInBackground(Unknown Source)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.support.v4.content.AsyncTaskLoader$LoadTask.doInBackground(Unknown Source)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.support.v4.content.AsyncTaskLoader$LoadTask.doInBackground(Unknown Source)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at android.support.v4.content.ModernAsyncTask$2.call(Unknown Source)
08-30 13:10:03.310: E/AndroidRuntime(14964):    at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
08-30 13:10:03.310: E/AndroidRuntime(14964):    ... 4 more

老实说,stacktrace 没有帮助,因为它不包含对我的代码的任何引用,而且这是一个非常奇怪的问题,因为如上所述,这两个片段在我第一次启动应用程序时工作,在手机布局上正常工作,但是屏幕旋转后我的平板电脑布局崩溃。

编辑:

在这两个片段中,我在 onActivityCreated 方法中加载数据,只需调用 getLoaderManager.initLoader(....);

这就是我创建装载机的方式:

第一个片段

@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {

    CursorLoader cursorLoader = new CursorLoader(getActivity(), MyContract.CONTENT_URI,
            null, null, null, null);

    return cursorLoader;
}

第二个片段:

@Override
public Loader<Cursor> onCreateLoader(int id, Bundle args) {

    return new CursorLoader(getActivity(), MyContract.CONTENT_URI, null,
            null, null, null);
}

这是我 CP 的内容 uri 的声明:

//MyContract class
public static final Uri CONTENT_URI = Uri.parse("content://" + MyProvider.AUTHORITY + "/items");

//In MyProvider class
public static final String AUTHORITY = "it.rciovati.mypackage.MyProvider"
4

3 回答 3

3

有一个错误在android.content.ContentResolver.acquireUnstableProvider()2013 年 11 月才被修复。可以在此处查看该错误的详细信息:

https://bugzilla.mozilla.org/show_bug.cgi?id=904551

本质上,如果 Uri 为 null 并且会导致崩溃,则该函数无法处理。为了正确支持较旧的 Android 版本,您需要确保永远不会传入 null Uri,或者捕获抛出的 NPE。

于 2014-02-17T18:13:02.240 回答
0

我解决了。在我的布局中,我有一个加载其他数据的第三个片段,并且对其进行了错误检查。

于 2012-08-30T15:39:50.023 回答
0

我也有同样的问题。事实证明,这是一个非常容易消除的错误。基于我的代码的示例

        Uri content_uri = null;

        if (bundle != null && bundle.containsKey("ID_BUNDLE_KEY")) {
            int word_id_to_query = bundle.getInt(ID_BUNDLE_KEY, 1);
            content_uri = ContentUris.withAppendedId(WordListContentProvider.CONTENT_URI, word_id_to_query);

        }

        cursorLoader = new CursorLoader(getActivity()
                .getApplicationContext(), content_uri,
                projections, SELECTION, SELECTION_ARGS, null);

我的错误是使用“ID_BUNDLE_KEY”作为参数而不是 ID_BUNDLE_KEY。后者已经是最终的静态字符串变量。因此 bundle.containsKey 将始终返回 false。这导致 content_uri 的变量在传递给新的 CurorLoader 函数时包含“null”。我的建议是在您自己的代码中验证“空”uri 没有错误地传入。

有关额外信息,Android 4.1.2 AOSP 代码中 ContentResolver.java 的第 1094 行确实是当 [uri.getscheme() != "content"] 的条件返回 false 时,导致提供者为空。运行时库抱怨的是这个 null 提供程序。

于 2013-06-27T17:27:47.537 回答