指定的孩子已经有一个父母。您必须先在孩子的父母上调用removeView()(Android)


164

我必须经常在两种布局之间切换。错误在下面发布的布局中发生。

第一次调用布局时,不会发生任何错误,一切都很好。然后,当我调用其他布局(空白)并随后再次调用我的布局时,它将引发以下错误:

> FATAL EXCEPTION: main
>     java.lang.IllegalStateException: The specified child already has a parent. You must call removeView() on the child's parent first.

我的布局代码如下所示:

    tv = new TextView(getApplicationContext()); // are initialized somewhere else
    et = new EditText(getApplicationContext()); // in the code


private void ConsoleWindow(){
        runOnUiThread(new Runnable(){

     @Override
     public void run(){

        // MY LAYOUT:
        setContentView(R.layout.activity_console);
        // LINEAR LAYOUT
        LinearLayout layout=new LinearLayout(getApplicationContext());
        layout.setOrientation(LinearLayout.VERTICAL);
        setContentView(layout);

        // TEXTVIEW
        layout.addView(tv); //  <==========  ERROR IN THIS LINE DURING 2ND RUN
        // EDITTEXT
        et.setHint("Enter Command");
        layout.addView(et);
        }
    }
}

我知道这个问题以前曾被问过,但对我来说并没有帮助。


1
仅适用于遇到相同错误的人:确保添加正确的元素。假设您必须添加,LinearLayout但您要添加TextView。所以解决它。
开发人员

当使用android数据绑定时,不应声明id为“ root”的视图,这会导致相同的错误。
Mohammad Reza Khahani

对于那些正在使用的用户TranstitionManager.beginDelayedTransition,请在这里
mochadwi

Answers:


354

错误消息说明您应该做什么。

// TEXTVIEW
if(tv.getParent() != null) {
    ((ViewGroup)tv.getParent()).removeView(tv); // <- fix
}
layout.addView(tv); //  <==========  ERROR IN THIS LINE DURING 2ND RUN
// EDITTEXT

56

只需将attachtoroot参数传递为false

View view = inflater.inflate(R.layout.child_layout_to_merge, parent_layout, false);

2
对我来说,这是一个靶心。recycleView中项目的约束布局,基于运行时动态更改约束。直到这一步,ConstraintSet.apply才起作用。因此,在onCreateViewHolder中传播父对象,但将false作为附加参数发送给膨胀。
miroslavign

3
这是正确的解决方案!正如@Sniper所说,将null传递给第二个参数可能会导致子视图无法与其父视图对齐。OTOH通过父级可能会导致相关错误。因此,attachToRoot= false是必经之路。
瓦西利斯

正确的做法。
罗兰·姆特特兹

50

我是在这里用我的recyclerview搜索错误的,但是解决方案没有用(显然)。在recyclerview的情况下,我已经写了原因和解决方法。希望它能帮助某人。

如果onCreateViewHolder()遵循以下方法,则会导致错误:

layoutInflater = LayoutInflater.from(context);
return new VH(layoutInflater.inflate(R.layout.single_row, parent));

相反,它应该是

return new VH(layoutInflater.inflate(R.layout.single_row, null));

15
有时随机答案(并非完全是对问题的答案)会帮助其他人。这对我有用。谢谢!
Ajith Memana

1
太好了!这就是为什么人们通常应该在充气机中将“ null”作为2d参数传递。谢谢。
超级用户

4
将Null传递给负责父级的参数并不是一个坏主意,但是您必须知道,在这种情况下,子视图(您膨胀的视图)在某些情况下将无法正确地自我测量,因为它不会对父母一无所知。在某些情况下,它会起作用,但在某些情况下,则不会。
Stoycho Andreev '18

1
请注意,这可能会导致无法正确解析Viewholders主题。
SpaceBison

13

我在尝试使用将根目录附加到true而不是false来提交片段时收到了此消息,如下所示:

return inflater.inflate(R.layout.fragment_profile, container, true)

完成后:

return inflater.inflate(R.layout.fragment_profile, container, false)

有效。


5

如果其他解决方案无效,例如:

View view = inflater.inflate(R.layout.child_layout_to_merge, parent_layout, false);

检查您从片段的onCreateView返回的是单视图还是视图组?在我的情况下,我在fragment xml的根目录上有viewpager,并且我正在返回viewpager,当我在布局中添加viewgroup时,我没有更新,我必须立即返回viewgroup,而不是viewpager(view)。


5

frameLayout.addView(bannerAdView); <-----如果您在此行上遇到错误,请执行以下操作。

if (bannerAdView.getParent() != null)

  ((ViewGroup) bannerAdView.getParent()).removeView(bannerAdView);

   frameLayout.addView(bannerAdView);     <------ now added view


3

就我而言,问题是由于我用<merge>布局夸大了父级View而引起的。在这种情况下,addView()导致崩溃。

View to_add = inflater.inflate(R.layout.child_layout_to_merge, parent_layout, true);
// parent_layout.addView(to_add); // THIS CAUSED THE CRASH

删除addView()有助于解决问题。



2

下面的代码为我解决了它:

@Override
public void onDestroyView() {
    if (getView() != null) {
        ViewGroup parent = (ViewGroup) getView().getParent();
        parent.removeAllViews();
    }
    super.onDestroyView();
}

注意:该错误来自我的片段类,并且通过覆盖这样的onDestroy方法,我可以解决它。



2

就我而言,当我想将父视图添加到其他视图时,会发生这种情况

View root = inflater.inflate(R.layout.single, null);
LinearLayout lyt = root.findViewById(R.id.lytRoot);
lytAll.addView(lyt);  // ->  crash

您必须添加这样的父视图

View root = inflater.inflate(R.layout.single, null);
LinearLayout lyt = root.findViewById(R.id.lytRoot);
lytAll.addView(root);

2

您必须首先从其父视图中删除子视图。

如果您的项目在Kotlin中,则您的解决方案看上去将与Java稍有不同。Kotlin使用简化了转换as?,如果左侧为null或转换失败,则返回null。

(childView.parent as? ViewGroup)?.removeView(childView)
newParent.addView(childView)

Kotlin扩展解决方案

如果您需要多次执行此操作,请添加此扩展名以使代码更具可读性。

childView.removeSelf()

fun View?.removeSelf() {
    this ?: return
    val parentView = parent as? ViewGroup ?: return
    parentView.removeView(this)
}

如果此View为null,父视图为null或父视图不是ViewGroup,它将安全地不执行任何操作


1

我找到了另一个解决方法:

if (mView.getParent() == null) {
                    myDialog = new Dialog(MainActivity.this);
                    myDialog.setContentView(mView);
                    createAlgorithmDialog();
                } else {
                    createAlgorithmDialog();
                }

在这里,我只是有一条if语句,用于检查视图是否具有父级,以及是否没有创建新对话框,请设置contentView并在我的“ createAlgorithmDialog()”方法中显示该对话框。

这也可以通过onClickListeners设置肯定和否定按钮(确定和取消按钮)。



0

我的情况是不同的,子视图已经有一个父视图,我正在将父视图内的子视图添加到另一个父视图。下面的示例代码

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_margin="@dimen/lineGap"
>
<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:textColor="@color/black1"
    android:layout_gravity="center"
    android:gravity="center"
    />

</LinearLayout>

然后我将这个视图放大并添加到另一个LinearLayout中,然后从上述布局中删除了LinaarLayout并开始工作

下面的代码解决了这个问题:

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:gravity="center"
   android:textColor="@color/black1" />

0

我的问题与许多其他答案有关,但是需要进行更改的原因略有不同...我试图将一个Activity转换为一个Fragment。因此,我将膨胀代码从onCreate移到了onCreateView,但是我忘记了将setContentView转换为inflate方法,并且相同的IllegalStateException也将我带到了此页面。

我改变了这个:

binding = DataBindingUtil.setContentView(requireActivity(), R.layout.my_fragment)

对此:

binding = DataBindingUtil.inflate(inflater, R.layout.my_fragment, container, false)

那解决了问题。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.