我有一个片段,片段A,它包含一个ViewPager。 ViewPager加载不同的片段,用户可以无限制地扫描#34;#34; (我使用非常多的页面/循环来模拟这个)。当用户单击当前的ViewPager片段时,片段管理器中的片段B将替换带有ViewPager的片段A.当用户从片段B返回时,使用popBackStackImmediate()
弹出backstack。如果用户多次重复此操作,则堆开始每次填充大约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并手动销毁它们?任何帮助都将非常感谢,谢谢。
答案 0 :(得分:1)
将此添加到Fragment会阻止我泄漏:
@Override
public void onDestroyView() {
super.onDestroyView();
viewPager.setAdapter(null);
}
查看源代码,问题似乎是在调用ViewPager#setAdapter
时,视图会将自身注册为适配器的观察者。因此,每次调用onViewCreated时,您的寻呼机适配器实例都将引用新创建的视图。
答案 1 :(得分:0)
根据您的需要有一个特定的PagerAdapter - FragmentStatePagerAdapter
当存在大量页面时,此版本的寻呼机更有用,更像列表视图。当页面对用户不可见时,它们的整个片段可能会被破坏,只保留该片段的保存状态。与FragmentPagerAdapter相比,这允许寻呼机保持与每个被访问页面相关联的更少的存储器,代价是在页面之间切换时可能具有更多的开销。