使用带有嵌套片段的 ViewPagerAdapter 时的 Android 内存泄漏问题

问题描述 投票:0回答:3

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

android android-viewpager fragmentmanager
3个回答
1
投票

将其添加到 Fragment 中可以阻止我的泄漏:

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

查看源代码,问题似乎是当调用

ViewPager#setAdapter
时,视图会将自身注册为适配器的观察者。因此,每次调用 onViewCreated 时,您的寻呼机适配器实例都会引用新创建的视图。


0
投票

有一个特定的 PagerAdapter 可以满足您的需求 - FragmentStatePagerAdapter

此版本的分页器在页面数量较多时更有用,更像列表视图。当页面对用户不可见时,其整个片段可能会被销毁,仅保留该片段的已保存状态。与 FragmentPagerAdapter 相比,这允许分页器保留与每个访问的页面关联的少得多的内存,但在页面之间切换时可能会产生更多的开销。


0
投票

我可以修复它的唯一方法是使用它提供的其他构造函数:

     adapter = MyPagerFragmentAdapter(
lifecycle = hostFragment.viewLifecycleOwner.lifecycle,
 fm = hostFragment.childFragmentManager
)
© www.soinside.com 2019 - 2024. All rights reserved.