2

我在 onStart() 方法中启动了一个 AsyncTask,因为我希望它在用户返回到该 Fragment 时始终从 SQLite 数据库重新加载数据。这是因为它是一个 ListFragment,并且单击一个项目会在平板电脑上的双窗格布局中显示另一个 Fragment,或者会在列表中的所有项目上使用 ViewPager 的 Activity。在后一种情况下,我需要重新加载项目,因为用户可以从 ViewPager 中删除项目。

现在我正在各种设备和 API 上测试我的布局。

在 Froyo (API 8) NexusOne AVD 上,如果我旋转屏幕一次,它会再次启动 AsyncTask 并重新显示数据。如果我再做一次,我会得到一个 NullPointerException。当我使用 API 10+ 启动另一个 AVD 时,不会发生这种情况(我可以随意旋转显示器)。我给它足够的时间来完成第一次旋转,然后再进行第二次旋转。

这是LogCat:

11-14 21:03:46.485: E/AndroidRuntime(326): FATAL EXCEPTION: main
11-14 21:03:46.485: E/AndroidRuntime(326): java.lang.NullPointerException
11-14 21:03:46.485: E/AndroidRuntime(326): at android.widget.ArrayAdapter.init(ArrayAdapter.java:271)
11-14 21:03:46.485: E/AndroidRuntime(326): at android.widget.ArrayAdapter.<init>(ArrayAdapter.java:150)
11-14 21:03:46.485: E/AndroidRuntime(326): at com.android.myapp.MyListFragment$MyAdapter.<init>(MyListFragment.java:279)
11-14 21:03:46.485: E/AndroidRuntime(326): at com.android.myapp.MyListFragment$getMyAsync.onPostExecute(MyListFragment.java:393)
11-14 21:03:46.485: E/AndroidRuntime(326): at com.android.myapp.MyListFragment$getMyAsync.onPostExecute(MyListFragment.java:1)
11-14 21:03:46.485: E/AndroidRuntime(326): at android.os.AsyncTask.finish(AsyncTask.java:417)
11-14 21:03:46.485: E/AndroidRuntime(326): at android.os.AsyncTask.access$300(AsyncTask.java:127)
11-14 21:03:46.485: E/AndroidRuntime(326): at android.os.AsyncTask$InternalHandler.handleMessage(AsyncTask.java:429)
11-14 21:03:46.485: E/AndroidRuntime(326): at android.os.Handler.dispatchMessage(Handler.java:99)
11-14 21:03:46.485: E/AndroidRuntime(326): at android.os.Looper.loop(Looper.java:123)
11-14 21:03:46.485: E/AndroidRuntime(326): at android.app.ActivityThread.main(ActivityThread.java:4627)
11-14 21:03:46.485: E/AndroidRuntime(326): at java.lang.reflect.Method.invokeNative(Native Method)
11-14 21:03:46.485: E/AndroidRuntime(326): at java.lang.reflect.Method.invoke(Method.java:521)
11-14 21:03:46.485: E/AndroidRuntime(326): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:868)
11-14 21:03:46.485: E/AndroidRuntime(326): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:626)
11-14 21:03:46.485: E/AndroidRuntime(326): at dalvik.system.NativeStart.main(Native Method)

MyListFragment.java:279 是调用ArrayAdapter .constructor [编辑]

super(context, 0, myListArray);

onPostExecute(MyListFragment.java:393 是这样的:

mAdapter = new MyAdapter(getActivity(), myListArray);

如果 myListArray 为空,则不会执行上述操作。

我在该指令上方添加了这个:

if (getActivity() == null) {
    Log.d(TAG, "getActivity() is null");
}

getActivity() 在 API 8 设备上返回 null!

这只是 Eclipse/ADT 和/或 AVD 测试环境的错误吗?或者它也会发生在真实的手机上?

谢谢。

编辑

启动 AsyncTask 的方法:

@Override
public void onStart() {
    super.onStart();
    mProgressBarLayout.setVisibility(View.VISIBLE);
    new getListAsync().execute(mType);
}

AsyncTask(ListFragment 的内部私有类):

    private class getListAsync extends AsyncTask<EFFECT_TYPE, Void, List<Item>> {

    @Override
    protected List<Item> doInBackground(EFFECT_TYPE... params) {
        List<Item> items;
        items = DatabaseManager.get(getActivity()).getItems(params[0]);
        return items;
    }

    @Override
    protected void onPostExecute(List<Item> result) {
        super.onPostExecute(result);
        mProgressBarLayout.setVisibility(View.GONE);
        mItemList = result;
        if (mItemList != null) {
            if (getActivity() == null) {
                Log.d(TAG, "getActivity() is null");  <---- THIS GETS LOGGED ON THE 2nd ROTATE!
            }
393 ----------->    mAdapter = new MyAdapter(getActivity(), mItemList);
            setListAdapter(mAdapter);
        }
    }

}

编辑#2

好吧,这是不可能的。

我已经有了 onAttach(),所以我这样做了:

向 MyListFragment 添加一个新变量:

private Activity mParentActivity = null;

修改 onAttach():

@Override
public void onAttach(Activity activity) {
    super.onAttach(activity);
    mCallbacks = (Callbacks) activity;
    mParentActivity = activity;
    if (mParentActivity == null) {
        Log.d(TAG, "onAttach() called with null Activity");
    }
}

onPost执行:

@Override
    protected void onPostExecute(List<Item> result) {
        super.onPostExecute(result);
        mProgressBarLayout.setVisibility(View.GONE);
        mItemList = result;
        if (mItemList != null) {
            if (mParentActivity  == null) {
                Log.d(TAG, "mParentActivity is null"); 
            }
            mAdapter = new MyAdapter(mParentActivity, mItemList);
            setListAdapter(mAdapter);
        }
    }

}

mParentActivity 在那里是 NULL !即使 onAttach() 在到达 onStart() 方法之前使用有效的 Activity 成功执行。

AsyncTask 在 API 8 上被破坏。

4

1 回答 1

0

这是我现在的解决方案。如果有人知道更好的,请发布它。

@Override
public void onStart() {
    super.onStart();
    mProgressBarLayout.setVisibility(View.VISIBLE);
    if (Build.VERSION.SDK_INT > Build.VERSION_CODES.FROYO) {
        new getItemsAsync().execute(mType);  // Do the AsyncTask for API 10+
    } else {
        // For FROYO only, do the operation on the main thread
        mItemsList = DatabaseManager.get(mParentActivity).getItems(mType);
        mProgressBarLayout.setVisibility(View.GONE);
        if (mItemsList != null) {
            mAdapter = new MyAdapter(mParentActivity, mItemsList);
            setListAdapter(mAdapter);
        }
    }
}

这行得通。

于 2013-11-15T06:54:09.903 回答