将ViewPagerAdapter与嵌套片段一起使用时,Android内存泄漏问题

时间:2016-02-08 11:58:06

标签: android android-viewpager fragmentmanager

我有一个片段,片段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并手动销毁它们?任何帮助都将非常感谢,谢谢。

2 个答案:

答案 0 :(得分:1)

将此添加到Fragment会阻止我泄漏:

@Override
public void onDestroyView() {
    super.onDestroyView();
    viewPager.setAdapter(null);
}

查看源代码,问题似乎是在调用ViewPager#setAdapter时,视图会将自身注册为适配器的观察者。因此,每次调用onViewCreated时,您的寻呼机适配器实例都将引用新创建的视图。

答案 1 :(得分:0)

根据您的需要有一个特定的PagerAdapter - FragmentStatePagerAdapter

  

当存在大量页面时,此版本的寻呼机更有用,更像列表视图。当页面对用户不可见时,它们的整个片段可能会被破坏,只保留该片段的保存状态。与FragmentPagerAdapter相比,这允许寻呼机保持与每个被访问页面相关联的更少的存储器,代价是在页面之间切换时可能具有更多的开销。