引言:这篇文章,记录了下FragmentTransaction的使用注意事项,和分析了下addToBackStack实际作用于谁。
public class MainActivity extends AppCompatActivity implements ItemFragment.OnListFragmentInteractionListener{
FragmentTransaction fragmentTransaction;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
ButterKnife.bind(this);
FragmentManager fragmentManager = getSupportFragmentManager();
fragmentTransaction = fragmentManager.beginTransaction();
fragmentTransaction.add(R.id.container, BlankFragment.newInstance("",""), "blank");
fragmentTransaction.add(R.id.container, BlankFragment2.newInstance("",""), "list");
// fragmentTransaction.addToBackStack("blank");
// fragmentTransaction.addToBackStack("list");
fragmentTransaction.addToBackStack("cccc");
fragmentTransaction.commit();
}
@Override
public void onListFragmentInteraction(DummyContent.DummyItem item) {
}
@OnClick({R.id.button1, R.id.button2})
protected void onClick(View v){
switch (v.getId()){
case R.id.button1:
getSupportFragmentManager().popBackStack();
break;
case R.id.button2:
Toast.makeText(this, "button2", Toast.LENGTH_LONG).show();
FragmentTransaction f = getSupportFragmentManager().beginTransaction();
f.replace(R.id.container, ItemFragment.newInstance(10), "list");
f.addToBackStack("list");
f.commit();
// fragmentTransaction.commit();
/* getSupportFragmentManager().beginTransaction().add(R.id.container, ItemFragment.newInstance(10), "list");
getSupportFragmentManager().beginTransaction().addToBackStack("list");
getSupportFragmentManager().beginTransaction().commit();*/
break;
}
}
}
1.FragmentTransaction注意的地方
1.每次fragmentManager.beginTransaction()得到的都是一个新实例。像下面对FragmentTransaction改动的代码是不会生效的。
getSupportFragmentManager().beginTransaction().add(R.id.container, ItemFragment.newInstance(10), "list");
getSupportFragmentManager().beginTransaction().addToBackStack("list");
getSupportFragmentManager().beginTransaction().commit();
2.每次对FragmentTransaction更改后需要commit来执行。但是这里有个要注意的地方就是,每个FragmentTransaction实例commit其实只能执行一次。如果将上图中的fragmentTransaction保存为全局后,当第一次commit后。在其他地方再去调用commit()就会报错(比如我在button2里去再次调用)。
protected void onCreate(Bundle savedInstanceState) {
...
fragmentTransaction.commit();
}
@OnClick({R.id.button1, R.id.button2})
protected void onClick(View v){
switch (v.getId()){
...
case R.id.button2:
...
fragmentTransaction.commit();
...
}
...
所以说第一次更改Fragment时,务必将所有的操作做完,再commit。
2.addToBackStack与popBackStack
addToBackStack其实针对的是fragmentTransaction。而非是具体的fragment。分析如下:(R.id.container是一个垂直的线性布局)从上图调用addToBackStack部分代码来看,我开始使用
fragmentTransaction.add(R.id.container, BlankFragment.newInstance("",""), "blank");
fragmentTransaction.add(R.id.container, BlankFragment2.newInstance("",""), "list");
fragmentTransaction.addToBackStack("list");
fragmentTransaction.commit();
当我点击button1,调用popBackStack()后,按照原先设想(addToBackStack是针对具体的fragment),那么应该是BlankFragment2被弹出,消失。但实际结果是BlankFragment和BlankFragment2都消失了。
当将addToBackStack("list")改成addToBackStack("blank"),结果依然是两个fragment都消失了。
这时我就怀疑,其实addToBackStack针对的是具体的FragmentTransaction。既然addToBackStack针对的是FragmentTransaction,那么他的参数tag其实跟Fragment的tag是没有任何关系的。所以我再次将addToBackStack("blank")改成addToBackStack("cccc")然后调用popBackStack()。发现结果依然是两个fragment都消失了。从这就可以看出来addToBackStack是针对的具体的FragmentTransaction。
3.addToBackStack与replace
我们都知道addToBackStack会将FragmentTransaction加入回退栈中,此时当前FragmentTransaction不再处于激活状态时,Fragment并不会被销毁。当我们想对同一容器叠加多个Fragment,实现动态添加新Fragment同时还能back退回到前一个Fragment。这个addToBackStack添加的位置是有讲究的。代码如下:
protected void onCreate(Bundle savedInstanceState) {
...
fragmentTransaction = fragmentManager.beginTransaction();
fragmentTransaction.addToBackStack("cccc");
fragmentTransaction.commit();
}
@OnClick({R.id.button1, R.id.button2})
protected void onClick(View v){
switch (v.getId()){
case R.id.button1:
getSupportFragmentManager().popBackStack();
...
case R.id.button2:
...
FragmentTransaction f = getSupportFragmentManager().beginTransaction();
f.replace(R.id.container, BlankFragment2.newInstance("第二层",""), "list");
// f.addToBackStack("list");
f.commit();
...
}
...
这样其实是不能实现该效果的。因为只是将第一层的FragmentTransaction管理的Fragment都加入了回退栈。同时因为调用replace,当我们点击button2,会看到第一层fragment不见了。这时候虽然fragmentTransaction管理的fragment没有被销毁但是视图其实都已经移除了。同时我们点击button1他其实是将fragmentTransaction移除掉,而不是f。所以我们正确的实现是将f加到回退栈。
f.addToBackStack(null);
这样后,会把第一层的fragment都去掉,然后添加第二层。点击button1调用popBackStack后,第一层又会回来(replace相当于remove和add的集合)。加入回退栈同时使用replace后,虽然第一层的实例没有销毁,但是视图已经是销毁了。如果还想保存第一层的视图,则把replace换成add。
如果对Fragment的一些基本概念还不是很了解,请看这篇文章 当我们谈Fragment时,我们谈些什么之一,这篇文章也不错Android -- Fragment 基本用法、生命周期与细节注意,也可以参考简析 addToBackStack使用和Fragment执行流程