为什么Fragment的onCreateView、onCreate、onActivityCreated方法会被调用?

26

我有一个应用程序,涉及到片段和ViewPager。在ViewPager中有三个片段。当你在它们之间切换时,它总是会导致另外两个片段调用它们的onCreateView方法。如何只在FragmentActivity创建时执行一次?

我阅读了一些问题并尝试了解决方案,但片段仍然具有相同的行为。

ListFragment onCreate called twice
onCreate() and onCreateView() invokes a lot more than required (Fragments)

以下是一些代码,如果对您有帮助:

MainActivity:

public class StartingActivity extends FragmentActivity implements View.OnClickListener {
ViewPager viewPager;
    CirclePageIndicator pageIndicator;

    Button discount;
    Button qrCode;
    Button pay;
    TabHost tabHost;

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.starting_layout);
        viewPager = (ViewPager) findViewById(R.id.pager);

        if (savedInstanceState == null) {

            Fragment firstPage = Fragment.instantiate(this, FindTovarFragment.class.getName());
            Fragment secondPage = Fragment.instantiate(this, MainWindowActivity.class.getName());
            Fragment thirdPage = Fragment.instantiate(this, MapActivity.class.getName());

            if ((firstPage != null && !firstPage.isDetached())|| (secondPage != null && !secondPage.isDetached()) || (thirdPage != null && !thirdPage.isDetached())) {

            List<Fragment> viewPagerFragments = new ArrayList<Fragment>();
            viewPagerFragments.add(firstPage);
            viewPagerFragments.add(secondPage);
            viewPagerFragments.add(thirdPage);


            PageAdapter pageAdapter = new PageAdapter(getSupportFragmentManager(), viewPagerFragments);

            viewPager.setAdapter(pageAdapter);

            pageIndicator = (CirclePageIndicator) findViewById(R.id.circle);
            pageIndicator.setViewPager(viewPager);
            pageIndicator.setCurrentItem(pageAdapter.getCount() - 2);
            }
        }

}

地图活动:

public class MapActivity extends Fragment implements OnMyLocationListener {

    //Тэг для логов
    private static final String TAG = "MapActivity";
    List<Address> addressList;
    private static final String STRING_LOCATION = "";

    ArrayList<TorgCentr> randomTorgCentr;
    ArrayList<String> torgCentrNames;

    Context context;
    AutoCompleteTextView searchTorgCentr;
    OverlayManager overlayManager;
    MapController mapController;
    TextView textView;
    double longitude;
    double latitude;
    double itemLongitude;
    double itemLatitude;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        Log.d(TAG, "MapActivity onCreateView");


        View view = (LinearLayout) inflater.inflate(R.layout.map_layout, container, false);
        final MapView mapView = (MapView) view.findViewById(R.id.map);
        textView = (TextView) view.findViewById(R.id.searchlocation);
        searchTorgCentr = (AutoCompleteTextView) view.findViewById(R.id.autoCompleteTextView);

        mapView.showBuiltInScreenButtons(true);
        mapController = mapView.getMapController();
        context = getActivity();
        return view;
    }

    @Override
    public void onResume() {
        super.onResume();
    }

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Log.d(TAG, "MapActivity onCreate");

    }

    public void onActivityCreated(Bundle savedInstanceState) {
        Log.d(TAG, "MapActivity onActivityCreated");
        context = getActivity();

        SetRightMapDisplayAddress rightMapDisplayAddress = new SetRightMapDisplayAddress();
        rightMapDisplayAddress.execute(STRING_LOCATION);

        DownloadSuperMarketsArray superMarketsArray = new DownloadSuperMarketsArray();
        superMarketsArray.execute();

        overlayManager = mapController.getOverlayManager();
        overlayManager.getMyLocation().setEnabled(false);

        super.onActivityCreated(savedInstanceState);
    }

第二片段:

public class MainWindowActivity extends Fragment {

    private static final String TAG = "MainWindowActivity";

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        Log.d(TAG, "MainWindowActivity onCreateView");
        View view = (RelativeLayout) inflater.inflate(R.layout.main_window_layout, container, false);
        if (container == null) {
            return null;
        }
        return view;
    }
}

第三个:

public class FindTovarFragment extends Fragment {

    private static final String TAG= "FindTovarFragment";

    Context context;
    ArrayList<Category> categories;
    Spinner categoryContainer;

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        Log.d(TAG, "FindTovarFragment onCreateView");

        View view = (LinearLayout) inflater.inflate(R.layout.find_tovar_main_layout, container, false);
        categoryContainer = (Spinner) view.findViewById(R.id.category);

        return view;
    }

    @Override
    public void onActivityCreated(Bundle savedInstanceState) {
        super.onActivityCreated(savedInstanceState);
        Log.d(TAG, "FindTovarFragment onActivityCreated");
        DownloadCategory downloadCategory = new DownloadCategory();
        downloadCategory.execute();
    }

地图活动日志:

06-20 11:06:37.709: DEBUG/MapActivity(1290): MapActivity onCreate
06-20 11:06:37.709: DEBUG/MapActivity(1290): MapActivity onCreateView
06-20 11:06:38.509: DEBUG/MapActivity(1290): MapActivity onActivityCreated

再一次,再一次:

06-20 11:07:53.239: DEBUG/MapActivity(1290): MapActivity onCreate
06-20 11:07:53.239: DEBUG/MapActivity(1290): MapActivity onCreateView
06-20 11:07:53.429: DEBUG/MapActivity(1290): MapActivity onActivityCreated
06-20 11:08:23.029: DEBUG/MapActivity(1290): MapActivity onCreate
06-20 11:08:23.039: DEBUG/MapActivity(1290): MapActivity onCreateView
06-20 11:08:23.269: DEBUG/MapActivity(1290): MapActivity onActivityCreated

非常感谢您的提前帮助。

3个回答

15

ViewPager默认在当前页面的左右保留1页以便快速切换,不会重新创建这些页面。但如果向左/右滑动超过1页,则会重新创建这些页面,因此调用OnCreateView()、OnCreate()方法。

如果应用程序使用了3页,则可以通过调用以下方法来增加要保留的页面数量:

mViewPager.setOffscreenPageLimit(2);

在这里描述 这里


12

我会根据Android开发文档上的建议,更改您的架构为以下架构:

http://developer.android.com/reference/android/support/v4/app/FragmentPagerAdapter.html

但我会对一些内容进行修改...

1- 我会更改这个方法:

/**
     * The Fragment's UI is just a simple text view showing its
     * instance number.
     */
    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container,
            Bundle savedInstanceState) {
        View v = inflater.inflate(R.layout.fragment_pager_list, container, false);
        View tv = v.findViewById(R.id.text);
        ((TextView)tv).setText("Fragment #" + mNum);
        return v;
    }

对于这样的情况,我们根据viewPager的位置决定要填充哪个碎片:

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,Bundle savedInstanceState) {
super.onCreateView(inflater, container, savedInstanceState);

SupportFragmentManager ft = getChildFragmentManager().beginTransaction();

String tag = "";
Fragment fragment = null;

  switch (mNum) {
  case 0:
    fragment = new MyFragmentZero();
    tag = FragmentTags.TAG_0;
    break;
  case 1:
    fragment = new MyFragmentOne();
    tag = FragmentTags.TAG_3;
    break;
  case 2:
    fragment = new MyFragmentTwo();
    tag = FragmentTags.TAG_2;
    break;
  default:
    break;
  }

/*OPTIONAL We can pass arguments to the fragments
Bundle args = new Bundle();
args.putInt(Arguments.ARG_POSITION, mNum);
fragment.setArguments(args);*/

//Place the fragment in the container
ft.replace(R.id.fragment_container fragment, tag);
ft.commit();

//You need a base layout for all fragment and use nested fragments later or you can define the layout for each position(mNum) inside the switch.
return inflater.inflate(R.layout.fragment_layout_default_for_all_views, container,
    false);
}

这样做可以拥有一个良好的架构,一旦它像这样工作就应该没问题了。

无论如何,你必须知道 viewPager 是如何在不同位置上填充 fragment 的。

当你从位置 0 开始时,位置 0 和位置 1 上的 fragment 就被创建了。

然后,当你滑动到位置 1 时,2 位置上的 fragment 被创建,因此现在你已经在不同的位置(0、1、2,假设 viewPager 上只有 3 页)上创建了三个 fragment。

我们向右滑到最后一个位置 2,第一个位置(0)上的 fragment 被销毁,因此现在我们只剩下位置 2 和 3 上的 fragment。

希望对你有所帮助,如果有问题请让我知道。干杯!


谢谢你的回答。但这正是我所做的。问题在于我的应用程序在创建片段时下载了大量信息并使用地理位置。因此,当一个片段被创建而另一个片段被视图页切换器销毁时,它必须重新下载所有内容,如果你快速滑动,应用程序会崩溃。所以,是否不可能自定义视图页切换器,使其不销毁对象? - PAcan
我认为不销毁片段不是一个好主意,这是一种非常糟糕的方法。你应该注意内存管理,不要这样做... - jpardogo
3
请勿重新下载所有内容!将此信息存储在您的活动中(您只需要将所需的内容下载到您的活动中一次),并通过片段界面与其通信。更改加载片段的方式,不要创建片段列表或类似的东西,只需要从适配器调用newInstance即可。请仔细阅读Android开发者文档,并遵循此代码,我认为这是最好的主意。将MainWindowActivity和MapActivity重命名为MainWindowFragment和MapFragment会更清晰明了。 - jpardogo
2
这是链接:http://developer.android.com/training/basics/fragments/communicating.html - jpardogo

11

最终我成功解决了问题。只需要覆盖destroyItem方法,以便它不销毁对象。希望这对某个人有帮助。

@Override
public void destroyItem(ViewGroup container, int position, Object object) {
    Log.d(TAG, "destroy!");
}

你能详细解释一下吗?你的销毁方法是什么样子的? - Simulant

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接