Fragment嵌套Fragment时遇到的那些坑

2016-07-21  本文已影响17096人  小木桨

由于项目要改成MVP模式,自然会用到了Fragment,有时候可能会需要一个Fragment里面嵌套多个Fragment,并且add完成后需要立即刷新子Fragment的View,那么这个时候就会抛出异常,先看一段代码:

MainActivity.java

 @Override
   protected void onCreate(Bundle savedInstanceState) {
   setContentView(R.layout.activity_main);
   //省略逻辑代码。。。
   mMessageFragment = MessageFragment.newInstance();
   getSupportFragmentManager()
                   .beginTransaction().add(R.id.layout_activity_main_fragment_contianer, mMessageFragment)
                   .commitAllowingStateLoss();
   }

MessageFragment.java

@Override
   public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
       View rootView = inflater.inflate(R.layout.fragment_message, null);
       ButterKnife.bind(this, rootView);
       return rootView;
   }

   @Override
   public void onActivityCreated(Bundle savedInstanceState) {
       super.onActivityCreated(savedInstanceState);
       initFragment();
       updateFragment();
   }
   
   private void updateFragment(){
       for(int i=0;i<mMessageFragments.length();i++){
           mMessageFragments[i].update();
       }
   }
   
   private void initFragment() {
       replyFragment = ConversationListFragment.newInstance(false);
replyFragment);
       privateFragment = ConversationListFragment.newInstance(true);
privateFragment);
       mMessageFragments = new Fragment[]{ replyFragment, privateFragment};
       getChildFragmentManager()
               .beginTransaction()
               .add(R.id.fragment_container, replyFragment)
               .add(R.id.fragment_container, privateFragment)
               .hide(privateFragment)
               .show(replyFragment)
               .commitAllowingStateLoss();
   }

ConversationListFragment.java

public static ConversationListFragment newInstance(boolean isprivate) {
       ConversationListFragment fragment = new ConversationListFragment();
       Bundle args = new Bundle();
       args.putBoolean("isprivate", isprivate);
       fragment.setArguments(args);
       return fragment;
   }

   @Override
   public void onCreate(Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       if (getArguments() != null) {
           mIsPrivate = getArguments().getBoolean("isprivate");
       }
   }

   @Nullable
   @Override
   public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
       View view = inflater.inflate(R.layout.fragment_message_conversation_list, null);
       ButterKnife.bind(this, view);
       return view;
   }
   
   public void update(){
   //当代码运行到这里的时候就会发生错误
   //比如:mIsPrivated的值是默认值,并没有获取到外部传过来的值,如果逻辑代码中需要用到View,则还会抛出空指针异常!
       if(mIsPrivated){
       //省略逻辑。。。
       }else{
       //省略逻辑。。。
       }
   }

分析原因

相信只要仔细看过上面的代码的都应该明白错误的原因了,主要原因就是对Fragment生命周期了解的不够透彻,因为在父Fragment的onActivityCreated方法中虽然实例化了子Fragment但是子Fragment的生命周期方法并没有被调用,所以这个时候子Fragment的View还没有被创建,如果直接在父Fragment的onActivityCreated方法中调用子Fragment更新View的方法就会抛出异常!

解决方案

既然知道了是因为子Fragment生命周期方法未执行引起的,那么就应该把更新子Fragmet的调用时机放到父Fragment的onActivCreated方法之外,可以放到父Fragment的onStart方法中即可!

总结

其实主要是对Fragment生命周期方法不熟造成的,再次回忆总结一下Fragment生命周期方法:


1354170699_6619.png

ActivityFragment对比图

1354170682_3824.png

生命周期分析:

  1. ragment被创建的时候
  1. fragment进入“后台模式”的时候
  1. fragment被销毁的时候
  1. 就像activitie一样,在以下的状态中,可以使用Bundle对象保存一个fragment的对象。
  1. fragments的大部分状态都和activitie很相似,但fragment有一些新的状态。
上一篇下一篇

猜你喜欢

热点阅读