【问题标题】:FragmentTabHost and Asynchronous TasksFragmentTabHost 和异步任务
【发布时间】:2014-08-06 10:17:17
【问题描述】:

问题:

我正在尝试创建两个包含 FragmentsFragmentTabHost 的选项卡。

这里是我将标签添加到FragmentTabHost 的地方(包含标签的FragmentActivityonCreate 方法:

@Override
protected void onCreate(Bundle savedInstanceState) {

    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_financials);

    mTabHost = (FragmentTabHost) findViewById(R.id.tabhost);
    mTabHost.setup(this, getSupportFragmentManager(), android.R.id.tabcontent);

    mTabHost.addTab(
            mTabHost.newTabSpec(TAG_ONE).setIndicator("Tab #1",
                    getResources().getDrawable(android.R.drawable.star_on)),
                    MyFragmentOne.class, null);
    mTabHost.addTab(
            mTabHost.newTabSpec(TAG_TWO).setIndicator("Tab #2",
                    getResources().getDrawable(android.R.drawable.star_on)),
                    MyFragmentTwo.class, null);


    if(savedInstanceState == null) {
        new MyAsyncTask(MyClass.this).execute();    
    }
    else {
        mTabHost.setCurrentTabByTag(savedInstanceState.getString("TAB_TAG"));
    }

}

但是,问题在于MyFragmentOneMyFragmentTwo 都在等待来自MyAsyncTask 的数据。即每个fragment中的视图只有在MyAsyncTask运行完成后才能正常创建。

因此,当我添加选项卡时,我最终会得到一个NullPointerException,因为AsyncTask 尚未获取所需的数据。

我的解决方案尝试:

我尝试在MyAsyncTaskonPostExecute() 中调用TabHost#addTab(),以便仅在 MyAsyncTask 完成后创建片段,从而避免任何空指针。

但是,通过将addTab() 调用移动到AsyncTaskonPostExecute(),我得到了错误:No tab tag for null。下面是 Logcat。

Logcat:

08-06 06:15:17.947: E/AndroidRuntime(1968): java.lang.IllegalStateException: No tab known for tag null
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.support.v4.app.FragmentTabHost.doTabChanged(FragmentTabHost.java:330)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.support.v4.app.FragmentTabHost.onAttachedToWindow(FragmentTabHost.java:280)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.View.dispatchAttachedToWindow(View.java:12585)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2458)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2465)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2465)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2465)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2465)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewGroup.dispatchAttachedToWindow(ViewGroup.java:2465)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1217)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1000)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5670)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.Choreographer$CallbackRecord.run(Choreographer.java:761)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.Choreographer.doCallbacks(Choreographer.java:574)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.Choreographer.doFrame(Choreographer.java:544)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:747)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.os.Handler.handleCallback(Handler.java:733)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.os.Handler.dispatchMessage(Handler.java:95)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.os.Looper.loop(Looper.java:136)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at android.app.ActivityThread.main(ActivityThread.java:5017)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at java.lang.reflect.Method.invokeNative(Native Method)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at java.lang.reflect.Method.invoke(Method.java:515)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:779)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:595)
08-06 06:15:17.947: E/AndroidRuntime(1968):     at dalvik.system.NativeStart.main(Native Method)

那么我该如何解决这个问题呢?

【问题讨论】:

    标签: android asynchronous android-fragments android-asynctask android-tabhost


    【解决方案1】:

    我最终使用了一个空白 PlaceHolderFragment 并以编程方式在 onPostExecute() 中删除它们,如下所示:

    mTabHost.addTab(
                mTabHost.newTabSpec("Placeholder one").setIndicator("Tab #1",
                        getResources().getDrawable(android.R.drawable.star_on)),
                        PlaceHolderFragment.class, null);
    mTabHost.addTab(
                mTabHost.newTabSpec("Placeholder two").setIndicator("Tab #2",
                        getResources().getDrawable(android.R.drawable.star_on)),
                        PlaceHolderFragment.class, null);
    

    然后在onPostExecute()

    mTabHost.setCurrentTab(0);
    mTabHost.clearAllTabs();
    
    mTabHost.addTab(
            mTabHost.newTabSpec(TAG_ONE).setIndicator("Tab #1",
                    getResources().getDrawable(android.R.drawable.star_on)),
                    MyFragmentOne.class, null);
    mTabHost.addTab(
            mTabHost.newTabSpec(TAG_TWO).setIndicator("Tab #2",
                    getResources().getDrawable(android.R.drawable.star_on)),
                    MyFragmentTwo.class, null);
    

    【讨论】:

    • 对不起,我看不到您的解决方案/ logcat 错误的任何逻辑。总之恭喜!! :)
    • 因为似乎我必须在onCreateView 中添加标签,所以我添加了两个空的虚拟标签。如果我不这样做,那么我会得到一个IllegalStateException。然后,在我的AsyncTask 中,一旦我从服务器获取所需的数据,它就会调用onPostExecute(),它会删除包含空白片段的虚拟选项卡,并添加我需要放在那里的两个选项卡。这样我也没有得到NullPointerException,因为AsyncTask已经获取了数据。
    • 如果您确实需要延迟加载 tabHost 内容,这似乎是唯一可能的解决方案。你实际上不需要添加两个虚拟选项卡,一个就足够了,但是从 onCreateView 返回视图后,tabHost 看起来不能为空
    【解决方案2】:

    您可以使用回调创建 AsyncTaks。然后在该回调中,您可以使用数据创建选项卡。

    例如: android asynctask sending callbacks to ui

    这只是一个例子,使用 asyncTask 和回调的例子很多

    【讨论】:

    • 回调仍然会导致相同的 no tab 标记为 null 错误,因为它本质上与将异步任务嵌套在片段活动中并在 onpostexecute 中实现它相同。 (我之前尝试过使用回调并得到同样的错误)
    • 好吧,你是对的。您是否尝试使用一些虚拟数据创建选项卡?
    • 是的,实际上我试过了,效果很好,我会把它作为答案发布。
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2013-02-10
    • 2017-09-20
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多