如何解决viewpager问题:指定的子组件已经有了父组件。您必须先在子组件的父组件上调用removeView()函数。

32

我的项目中使用了带有三个选项卡的viewpager,分别命名为历史主页地图主页包含计时器、秒表等功能。 地图显示Google地图。而在历史中,我目前只使用了简单的文本视图。

Viewpager流向: 历史 - 主页 - 地图

我把主页设为当前项目(默认情况下会显示主页标签)。现在,当我从主页滑动到地图或从地图滑动到主页时,一切都运行得很完美。但是,当我从主页滑动到历史时,虽然没有错误,但从历史返回到主页历史 > 主页)时,Eclipse会报错:

E/AndroidRuntime(  533): java.lang.IllegalStateException: The specified child already has a parent. You must call removeView() on the child's parent first.

顺便说一下,我正在使用 intent into Map Activity。请检查以下代码。请告诉我如何解决我的问题。

ViewPager 类:

@Override
    public Fragment getItem(int position) {
        // return SwipeyTabFragment.newInstance(TITLES[position]);

        Fragment f = new Fragment();

        switch (position) {
        case 0:
            f = History.newInstance(position);
            break;
        case 1:
            f = Main.newInstance(position);
            break;
        case 2:
            f = Map.newInstance(position);
            break;

        }

        return f;

    }

History.class:

public class History extends Fragment {

public static Fragment newInstance(int position) {
    History f = new History();
    Bundle args = new Bundle();

    args.putInt("title", position);
    f.setArguments(args);
    return f;

}

 @Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
    ViewGroup root = (ViewGroup) inflater.inflate(R.layout.history, null);


     ( (TextView) root.findViewById(R.id.text)).setText("Hello");
    return root;
} 

}

Main.class :

 public class Main extends Fragment implements GPSCallback {

 ......

public static Fragment newInstance(int position) {
    Main f = new Main();
    Bundle args = new Bundle();

    args.putInt("title", position);
    f.setArguments(args);
    return f;

}

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

    final ViewGroup root = (ViewGroup) inflater
            .inflate(R.layout.main, null);

    .......

    return root;
 }

    ........
}

Map.class:

public class Map extends Fragment {

private static final String KEY_STATE_BUNDLE = "localActivityManagerState";

private LocalActivityManager mLocalActivityManager;

protected LocalActivityManager getLocalActivityManager() {
    return mLocalActivityManager;
}

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    Bundle state = null;
    if (savedInstanceState != null) {
        state = savedInstanceState.getBundle(KEY_STATE_BUNDLE);
    }

    mLocalActivityManager = new LocalActivityManager(getActivity(), true);
    mLocalActivityManager.dispatchCreate(state);
}



public static Fragment newInstance(int position) {
    Map f = new Map();
    Bundle args = new Bundle();

    args.putInt("title", position);
    f.setArguments(args);
    return f;

}


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

        Intent i = new Intent(getActivity(), MapViewActivity.class);
       // Intent i = new Intent(getActivity(), hellogooglemap.class);
        Window w = mLocalActivityManager.startActivity("tag", i); 

        View currentView=w.getDecorView(); 
        currentView.setVisibility(View.VISIBLE); 
        currentView.setFocusableInTouchMode(true); 
        ((ViewGroup) currentView).setDescendantFocusability(ViewGroup.FOCUS_AFTER_DESCENDANTS);
        return currentView;


      /*ViewGroup root = (ViewGroup) inflater.inflate(R.layout.history, null);


         ( (TextView) root.findViewById(R.id.text)).setText("Hello");
        return root;*/

}


 @Override
    public void onSaveInstanceState(Bundle outState) {
        super.onSaveInstanceState(outState);
        outState.putBundle(KEY_STATE_BUNDLE,
                mLocalActivityManager.saveInstanceState());
    }

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

    @Override
    public void onPause() {
        super.onPause();
        mLocalActivityManager.dispatchPause(getActivity().isFinishing());
    }

    @Override
    public void onStop() {
        super.onStop();
        mLocalActivityManager.dispatchStop();
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        mLocalActivityManager.dispatchDestroy(getActivity().isFinishing());
    }

}

日志记录:

 E/AndroidRuntime(  533): java.lang.IllegalStateException: The specified child already has a parent. You must call removeView() on the child's parent first.
 E/AndroidRuntime(  533):   at android.view.ViewGroup.addViewInner(ViewGroup.java:1976)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.addView(ViewGroup.java:1871)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.addView(ViewGroup.java:1828)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.addView(ViewGroup.java:1808)
 E/AndroidRuntime(  533):   at android.support.v4.app.NoSaveStateFrameLayout.wrap(NoSaveStateFrameLayout.java:40)
 E/AndroidRuntime(  533):   at android.support.v4.app.FragmentManagerImpl.moveToState(FragmentManager.java:874)
 E/AndroidRuntime(  533):   at android.support.v4.app.FragmentManagerImpl.attachFragment(FragmentManager.java:1240)
 E/AndroidRuntime(  533):   at android.support.v4.app.BackStackRecord.run(BackStackRecord.java:612)
 E/AndroidRuntime(  533):   at android.support.v4.app.FragmentManagerImpl.execPendingActions(FragmentManager.java:1416)
 E/AndroidRuntime(  533):   at android.support.v4.app.FragmentManagerImpl.executePendingTransactions(FragmentManager.java:431)
 E/AndroidRuntime(  533):   at android.support.v4.app.FragmentPagerAdapter.finishUpdate(FragmentPagerAdapter.java:139)
 E/AndroidRuntime(  533):   at android.support.v4.view.ViewPager.populate(ViewPager.java:804)
 E/AndroidRuntime(  533):   at android.support.v4.view.ViewPager.completeScroll(ViewPager.java:1280)
 E/AndroidRuntime(  533):   at android.support.v4.view.ViewPager.computeScroll(ViewPager.java:1176)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.drawChild(ViewGroup.java:1562)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1373)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.drawChild(ViewGroup.java:1644)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1373)
 E/AndroidRuntime(  533):   at android.view.View.draw(View.java:6883)
 E/AndroidRuntime(  533):   at android.widget.FrameLayout.draw(FrameLayout.java:357)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.drawChild(ViewGroup.java:1646)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1373)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.drawChild(ViewGroup.java:1644)
 E/AndroidRuntime(  533):   at android.view.ViewGroup.dispatchDraw(ViewGroup.java:1373)
 E/AndroidRuntime(  533):   at android.view.View.draw(View.java:6883)
 E/AndroidRuntime(  533):   at android.widget.FrameLayout.draw(FrameLayout.java:357)
 E/AndroidRuntime(  533):   at com.android.internal.policy.impl.PhoneWindow$DecorView.draw(PhoneWindow.java:1862)
 E/AndroidRuntime(  533):   at android.view.ViewRoot.draw(ViewRoot.java:1522)
 E/AndroidRuntime(  533):   at android.view.ViewRoot.performTraversals(ViewRoot.java:1258)
 E/AndroidRuntime(  533):   at android.view.ViewRoot.handleMessage(ViewRoot.java:1859)
 E/AndroidRuntime(  533):   at android.os.Handler.dispatchMessage(Handler.java:99)
 E/AndroidRuntime(  533):   at android.os.Looper.loop(Looper.java:130)
 E/AndroidRuntime(  533):   at android.app.ActivityThread.main(ActivityThread.java:3683)
 E/AndroidRuntime(  533):   at java.lang.reflect.Method.invokeNative(Native Method)
 E/AndroidRuntime(  533):   at java.lang.reflect.Method.invoke(Method.java:507)
 E/AndroidRuntime(  533):   at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:839)
 E/AndroidRuntime(  533):   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:597)
 E/AndroidRuntime(  533):   at dalvik.system.NativeStart.main(Native Method)
 W/ActivityManager(   68):   Force finishing activity      com.android.gps/.viewpager.SwipeyTabsSampleActivity
 W/ActivityManager(   68): Activity pause timeout for HistoryRecord{40716740 com.android.gps/.viewpager.SwipeyTabsSampleActivity}

谢谢。

更新:

如果不进入地图活动,它就可以完美运行。我的意思是当我将其用作历史记录时,进入地图没有任何错误。

9个回答

41

当我使用时,我遇到了相同的问题

View res = inflater.inflate(R.layout.fragment_guide_search, container);

在 Fragment.onCreateView(...) 中:

你必须调用

View res = inflater.inflate(R.layout.fragment_guide_search, container, false);
或者
View res = inflater.inflate(R.layout.fragment_guide_search, null);

根据文档的描述,这是非常合理的,并且解决了我的问题。 - CoatedMoose
1
View res = inflater.inflate(R.layout.fragment_guide_search, null); 视图 res = inflater.inflate(R.layout.fragment_guide_search, null); - cV2
对我来说,第二个选项也可以。 - androidevil
1
请注意,第二个选项将忽略根充气视图上的布局参数;如果您有容器视图可用,则第一个选项(容器,false)几乎始终是正确的选项。 - Karu
运行完美!非常感谢 :) - Kiril Aleksandrov

27

我也遇到了这个问题。

你可以通过添加一行代码 mViewPager.setOffscreenPageLimit(3); 来解决它。

public class SwipeyTabsSampleActivity extends FragmentActivity {

...

@Override
public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    setContentView(R.layout.main);

    mViewPager = (ViewPager) findViewById(R.id.viewpager);
    mTabs = (SwipeyTabs) findViewById(R.id.swipeytabs);

    SwipeyTabsPagerAdapter adapter = new SwipeyTabsPagerAdapter(this,
            getSupportFragmentManager());
    mViewPager.setAdapter(adapter);

    mViewPager.setOffscreenPageLimit(3);  <------  Add this one
}

}

祝你好运。


这并不解释崩溃的原因。通常,这是因为在片段“onCreateView”中返回了错误的视图。在这个问题中,最有可能的是Map.java的onCreateView有问题。 - Aura Lee

4

首先,RB Patel提供的插入代码的答案(如下所示)非常有效,但是当我了解到这个功能时,我意识到它旨在作为潜在的优化,实际上并不是解决该错误的方法。

mViewPager.setOffscreenPageLimit(3);

安卓开发者网站对此方法的解释如下:
setOffscreenPageLimit(int limit) - 设置在空闲状态下视图层次结构中当前页面一侧应保留的页面数。
看起来这只是设置了缓存到当前页左侧或右侧的页面数量限制。在我的情况下,这只是问题的紧急措施,是的错误停止了,并且事情变得顺利了,但是我仍然有一个坏代码潜藏在我的程序中,没有被执行。我相信这种方法旨在用作优化而不是修复错误的实际方法。
当我尝试重载扩展PagerAdapter的自定义类中的instantiateItem()方法时,我遇到了这个问题。在我回到之前访问/实例化的选项卡/页面后,我的错误被抛出。
我是安卓的新手,但我会尽力解释我的解决方案。引发此错误的行是以下行:
v.addView(child);

如果我理解正确,出现问题是因为该子项之前已经被实例化并分配了父项。
当我回到以前访问过的视图/页面时,我试图将一个已经有父布局视图的视图添加到新布局中,因此出现了错误。
解决方案:我只是检查了一下我尝试显示(实例化)的子视图是否已经在其他地方实例化,方法是看它是否有设置父视图。如果子视图已经被创建,则我只使用了子视图的父项而不是创建新布局。
以下是我的代码(如果有任何专家感到冒犯,请原谅):
@Override
public Object instantiateItem(View container, int position) { 
    LinearLayout v = new LinearLayout(mContext);

    //the view being loaded (unless it already was then we have to handle it)
    View child = mViews.get(position);

    //If the child view was loaded already then it will have a parent view it belongs to
    //We return the parent view instead of adding the child to the LinearLayout we just created and returning that new layout.
    if(child.getParent() != null) {
        View parent = (View) child.getParent();
        ((ViewPager) container).addView(parent);
        return parent;
    }

    //first time instantiating a child view/page
    v.addView(child);            

    // These lines execute the first time a given page is instantiated
    ((ViewPager) container).addView(v);

    return v;
}

mViewPager.setOffscreenPageLimit(3); 这句代码救了我的命!非常感谢! - L93
返回父级仍然出现异常。 - Allen Vork

2

// 如果要添加的子视图已经绑定到父视图,则将该子视图从父视图中移除,然后重新添加到容器中。

if (v.getParent() != null) {
  ((ViewGroup) v.getParent()).removeView(v);
}
container.addView(v);

如果您想在onCreate()方法之后执行某些操作,那么这就是正确的解决方案。您不需要adapter.setOffScreenPageLimit(int)。谢谢Richa。 - Khalid Lakhani

2
如果您正在使用tabLayout或viewPager,对于一个adapter,替换getChildFragmentManager()getActivity().getSupportFragmentManager(),如下所示。 TabAdapter adapter = new TabAdapter(getActivity(), getActivity().getSupportFragmentManager());

当其他所有方法都失败时,这就是我的救命稻草。同时,对于任何阅读此内容的人,请确保检查 getActivity() 是否为 null,因为在设备旋转或活动重新启动(例如用户更改应用程序主题)期间可能会发生这种情况。 - Shadow

1
这是我的解决方案(在每个片段中都使用),它既可以保证流畅性,又可以避免内存问题。
...
private LayoutInflater mInflater;
private WeakReference<View> mRootView = null;
...

@Override
public View onCreateView(final LayoutInflater inflater, final ViewGroup container, final Bundle savedInstanceState) 
  {
  if (inflater != null)
    mInflater = inflater;
  else
    mInflater = LayoutInflater.from(getActivity());
  View rootView = mRootView == null ? null : mRootView.get();
  if (rootView != null) 
      {
      final ViewParent parent = rootView.getParent();
      if (parent != null && parent instanceof ViewGroup)
        ((ViewGroup) parent).removeView(rootView);  
      }
  else 
       {
       rootView = mInflater.inflate(R.layout.fragment_test, null, false);
       mRootView = new WeakReference<View>(rootView);
       }
  return rootView;
  }

1
@Override
public Object instantiateItem(View arg0, int arg1) {
    Log.d("instantiateItem", ""+arg0+" "+arg1);
    try { 
        if(mListViews.get(arg1).getParent()==null)
            ((ViewPager) arg0).addView(mListViews.get(arg1), 0);  
        else{
            // I am new to android, it is strange that the view to be added is already bound to a parent
            // Through trials and error I solve this problem with the following codes
            // Add that the element of mlistviews is listview in pagerview;
            ((ViewGroup)mListViews.get(arg1).getParent()).removeView(mListViews.get(arg1));

            ((ViewPager) arg0).addView(mListViews.get(arg1), 0); 
        }
    } catch (Exception e) {   
        Log.d("parent=", ""+mListViews.get(arg1).getParent()); 
        e.printStackTrace();  
    }  
    return mListViews.get(arg1);
}

0
如果您正在使用FragmentPagerAdapter,则只需在FragmentPagerAdapter中使用destroyItem方法即可。
@Override
public void destroyItem(ViewGroup container, int position, Object object) 
{
        FragmentTransaction ft=fm.beginTransaction();
        ft.remove((Fragment) object);
        ft.commit();
}

setOffscreenPageLimit(int limit) 方法会消耗更多的内存,在某些情况下可能无法完美地工作。 它在任一侧都保留状态,请参见说明。setOffscreenPageLimit(int limit) - 在空闲状态下设置应保留在视图层次结构中当前页面两侧的页面数。


-1

检查一下你的布局文件中是否有任何ConstrainLayout(或类似)的内容。这可能是抛出该错误的原因。


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