我有一个片段,片段A,其中包含一个ViewPager。 ViewPager 加载不同的片段,用户可以“无限地”滑动这些片段(我使用大量的页面/循环来模拟这一点)。当用户单击当前 ViewPager 片段时,带有 ViewPager 的片段 A 会被片段管理器中的片段 B 替换。当用户从片段 B 返回时,使用
popBackStackImmediate()
弹出后台堆栈。如果用户多次重复此操作,堆就会开始一次填满约 100kb,直到内存填满时应用程序开始变得马虎并出现故障。我不确定到底是什么原因造成的,有人可以帮忙吗?
我的片段 A 与 ViewPager:
public class MainFragment extends Fragment {
private MainWearActivity mMainWearActivity;
View view;
private int currentPage;
private ViewPager pager;
private ViewPagerAdapter adapter;
private LinearLayout helpIcons;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mMainWearActivity = (MainWearActivity) getActivity();
adapter = new ViewPagerAdapter(this.getChildFragmentManager());
}
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
view = inflater.inflate(R.layout.fragment_main, container, false);
// Scrolling menu
pager = (ViewPager) view.findViewById(R.id.watchNavPager);
pager.setAdapter(adapter);
pager.addOnPageChangeListener(adapter);
// Set current item to the middle page
pager.setCurrentItem(Consts.FIRST_PAGE);
currentPage = Consts.FIRST_PAGE;
// Set number of pages
pager.setOffscreenPageLimit(4);
// Set no margin so other pages are hidden
pager.setPageMargin(0);
return view;
}
@Override
public void onDestroyView() {
pager = null;
super.onDestroyView();
}
}
我的适配器类:
public class ViewPagerAdapter extends FragmentPagerAdapter implements
ViewPager.OnPageChangeListener {
public ViewPagerAdapter(FragmentManager fm) {
super(fm);
}
@Override
public Fragment getItem(int position)
{
position = position % Consts.PAGES;
switch(position){
case Consts.AUDIO_POS:
return new AdapterAudioFragment();
case Consts.VOICE_POS:
return new AdapterVoiceFragment();
case Consts.MAIL_POS:
return new AdapterMailFragment();
case Consts.INFO_POS:
return new AdapterInfoFragment();
default:
return null;
}
}
@Override
public int getCount()
{
return Consts.PAGES * Consts.LOOPS; // (4 * 1000)
}
@Override
public void onPageScrolled(int position, float positionOffset,
int positionOffsetPixels) {}
@Override
public void onPageSelected(int position) {}
@Override
public void onPageScrollStateChanged(int state) {}
}
适配器加载的我的片段之一(它们几乎相同):
public class AdapterAudioFragment extends Fragment {
private ImageView menuImg;
private TextView menuText;
private LinearLayout rootView;
private MainWearActivity mMainWearActivity;
private View.OnClickListener imgClickListener;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
mMainWearActivity = (MainWearActivity) getActivity();
imgClickListener = new View.OnClickListener() {
@Override
public void onClick(View v) {
mMainWearActivity.replaceFragment(mMainWearActivity.getFragment(Consts.FRAG_AUDIO), Consts.FRAG_AUDIO);
}
};
}
@Override
public View onCreateView(LayoutInflater inflater,
ViewGroup container, Bundle savedInstanceState) {
// Get root view of the fragment layout
rootView = (LinearLayout) inflater.inflate(
R.layout.fragment_nav_object, container, false);
// Set the current menu image and text
menuImg = (ImageView) rootView.findViewById(R.id.fragment_image);
menuImg.setImageResource(R.mipmap.ic_audio);
menuImg.setOnClickListener(imgClickListener);
menuText = (TextView) rootView.findViewById(R.id.menuTxt);
menuText.setText(Consts.MENU_HEADER_AUDIO);
// Set the current menu selection
mMainWearActivity.setCurrentSelection(Consts.AUDIO_POS);
return rootView;
}
}
我有一种感觉,适配器的片段都被创建但从未被销毁并堆积在堆中,但我不知道如何解决这个问题。我需要在适配器中调用 destroyItem 并手动销毁它们吗?任何帮助将不胜感激,谢谢。
将其添加到 Fragment 中可以阻止我的泄漏:
@Override
public void onDestroyView() {
super.onDestroyView();
viewPager.setAdapter(null);
}
查看源代码,问题似乎是当调用
ViewPager#setAdapter
时,视图会将自身注册为适配器的观察者。因此,每次调用 onViewCreated 时,您的寻呼机适配器实例都会引用新创建的视图。
有一个特定的 PagerAdapter 可以满足您的需求 - FragmentStatePagerAdapter
此版本的分页器在页面数量较多时更有用,更像列表视图。当页面对用户不可见时,其整个片段可能会被销毁,仅保留该片段的已保存状态。与 FragmentPagerAdapter 相比,这允许分页器保留与每个访问的页面关联的少得多的内存,但在页面之间切换时可能会产生更多的开销。
我可以修复它的唯一方法是使用它提供的其他构造函数:
adapter = MyPagerFragmentAdapter(
lifecycle = hostFragment.viewLifecycleOwner.lifecycle,
fm = hostFragment.childFragmentManager
)