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

当我的布局第一次被调用时,没有发生任何错误,一切都很好。当我然后调用一个不同的布局(一个空白的),然后调用我的布局第二次,它抛出以下错误:

> 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);
        }
    }
}

我知道以前有人问过这个问题,但它对我的情况没有帮助。


错误消息告诉您应该做什么。

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

我来到这里用我的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));

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

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

在这里,我只是有一个if语句检查视图是否有一个父,如果它没有创建新的对话框,设置contentView并在我的“createAlgorithmDialog()”方法中显示对话框。

这也设置了积极和消极按钮(确定和取消按钮)与onClickListeners。


在我的情况下,问题是由我正在膨胀父视图与<合并>布局造成的。在本例中,是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()有助于解决这个问题。


简单地传递参数

随员=假

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

你可以使用这个方法来检查一个视图是否有子视图。

public static boolean hasChildren(ViewGroup viewGroup) {
    return viewGroup.getChildCount() > 0;
 }

如果其他解决方案不奏效,比如:

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

检查你从片段的onCreateView返回了什么它是单个视图还是视图组?在我的情况下,我在碎片的xml根上有viewpager,我正在返回viewpager,当我在布局中添加viewgroup时,我没有更新,我现在必须返回viewgroup,而不是viewpager(view)。


当我试图使用attach to root将一个片段提交到true而不是false时,我得到了这个消息,就像这样:

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

后做的事情:

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

它工作。


检查是否已经添加了视图

if (textView.getParent() == null)
    layout.addView(textView);

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

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

注意:这个错误来自我的片段类,通过重写onDestroy方法,我可以解决它。


frameLayout.addView (bannerAdView);<-----如果你在这一行得到错误,做如下..

if (bannerAdView.getParent() != null)

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

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

if(tv!= null){
    ((ViewGroup)tv.getParent()).removeView(tv); // <- fix
}

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

<?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" />

我的错误是这样定义视图:

view = inflater.inflate(R.layout.qr_fragment, container);

它不见了:

view = inflater.inflate(R.layout.qr_fragment, container, false);

在我的情况下,它发生时,我想添加视图由父母到其他视图

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);

首先必须从父视图中移除子视图。

如果您的项目是在Kotlin中,那么您的解决方案将与Java略有不同。Kotlin使用as?,如果左侧为空或强制转换失败,则返回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)
}

如果这个视图为空,父视图为空,或者父视图不是ViewGroup,它将安全地不做任何事情


我的问题与许多其他答案有关,但需要做出改变的原因略有不同……我试图将一个活动转换为一个片段。所以我把膨胀代码从onCreate移动到onCreateView,但我忘记从setContentView转换到膨胀方法,同样的IllegalStateException把我带到了这个页面。

我改了这个:

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

:

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

这就解决了问题。


当我为活动和片段使用数据绑定时,它发生在我身上。

对于fragment - in onCreateView,我们可以用传统的方式使用膨胀器来膨胀布局。 在onViewCreated方法中,绑定对象可以更新为

 binding = DataBindingUtil.getBinding<FragmentReceiverBinding>(view) as FragmentReceiverBinding

它解决了我的问题


就我而言,我做错了:

...
TextView content = new TextView(context);
for (Quote quote : favQuotes) {
  content.setText(quote.content);
...

代替(好):

...
for (Quote quote : favQuotes) {
  TextView content = new TextView(context);
  content.setText(quote.content);
...

如果在你的XML中,你有id为root的布局,这是有问题的,只是改变id名称


我也犯了同样的错误,看看我做了什么。我的坏,我试图将相同的视图NativeAdView添加到多个FrameLayouts,通过为每个FrameLayout创建一个单独的视图NativeAdView来解决,谢谢


在我的情况下,我不小心从Layout.onCreateView()中返回一个子视图,如下所示:

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
    View v= inflater.inflate(R.layout.activity_deliveries, container, false);
    RecyclerView rv  = (RecyclerView) v.findViewById(R.id.deliver_list);
    
    return rv; // <- here's the issue
}

解决方案是返回父视图(v)而不是子视图(rv)。


在KOTLIN中简化

 viewToRemove?.apply {
            if (parent != null) {
                (parent as ViewGroup).removeView(this)
            }
        }

在我的例子中,我有一个适配器,它与一个recyclerView一起工作,传递给适配器的项是具有自己视图的项。

所需要的只是一个线性布局作为每个项目传递的容器,所以我所做的是在onBindViewHolder内的指定位置抓取项目,然后将其添加到线性布局,然后显示。

在文档中检查基础知识,

当一个项目滚动出屏幕时,RecyclerView不会销毁它 视图

因此,与我的项目,当我滚动到一个方向,然后改变到相反的方向-快速,最近显示的项目还没有被销毁,这意味着,项目仍然与线性布局容器相关联,然后在我的端,我试图附加到另一个容器,这最终与一个孩子已经有一个父母。

我的解决方案是检查指定的项是否有父项,如果有,我将其分配给一个变量,然后调用parentVar.removeView(item),然后分配新的父项。

以下是示例代码(有问题的适配器):

override fun onBindViewHolder(holder: QuestionWidgetViewHolder, position: Int) {

    holder.linearLayoutContainer.removeAllViewsInLayout()

    val questionWidget: QuestionWidget =
        dataSource[position]

    questionWidget.setValueChangedListener(this)

    holder.linearLayoutContainer.addView(questionWidget)/*addView throws error once in a while*/
    

}


inner class QuestionWidgetViewHolder(mView: View) : 

RecyclerView.ViewHolder(mView) {
        val linearLayoutContainer: LinearLayout =
            mView.findViewById(R.id.custom_question_widget_container)

}

R.id.custom_question_widget_container的内容:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/custom_question_widget_container"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_margin="10dp"
    android:orientation="vertical"
    android:padding="10dp" />

所以,questionWidget似乎已经在可见性之外保留了几乎4步的父,当我快速滚动到相反的方向时,我会遇到它仍然与它的父,然后我试图将它添加到另一个容器。

这里是解决方案-选项1:

        override fun onBindViewHolder(holder: QuestionWidgetViewHolder, position: Int) {
    
            holder.linearLayoutContainer.removeAllViewsInLayout()
    
            val questionWidget: QuestionWidget =
                dataSource[position]
    
            questionWidget.setValueChangedListener(this)
    
       val initialWidgetParent : ViewParent? = questionWidget.parent

//attempt to detach from previous parent if it actually has one
        (initialWidgetParent as? ViewGroup)?.removeView(questionWidget)

        holder.linearLayoutContainer.addView(questionWidget)
            
    
        }

另一个更好的解决方案-选项2:

        override fun onBindViewHolder(holder: QuestionWidgetViewHolder, position: Int) {
    
            holder.linearLayoutContainer.removeAllViewsInLayout()
    
            val questionWidget: QuestionWidget =
                dataSource[position]
    
            questionWidget.setValueChangedListener(this)
    
        val initialWidgetParent : ViewParent? = questionWidget.parent
//if it's in a parent container already, just ignore adding it to a view, it's already visible

       if(initialWidgetParent == null) {
           holder.linearLayoutContainer.addView(questionWidget)
       }
            
    
        }

实际上,在将它添加到父节点之前,它需要询问子节点是否有父节点。


你只需要传递attachtorroot参数false。

mBinding = FragmentCategoryBinding.inflate(inflater, container, false)

我尝试了你们建议的所有方法,但都没有成功。

但是,我设法修复它通过移动我所有的绑定初始化从onCreate到onCreateView。

onCreate(){
        binding = ScreenTicketsBinding.inflate(layoutInflater)
}

搬到

onCreateView(...){
            binding = ScreenTicketsBinding.inflate(layoutInflater) 
}

在我的例子中,我将id命名为约束布局的“root”,这与现有的父根id冲突。

试着改变id。

<androidx.constraintlayout.widget.ConstraintLayout
            android:id="@+id/root"  //<--It should not named as root.
            android:layout_width="match_parent"
            android:layout_height="match_parent"

</androidx.constraintlayout.widget.ConstraintLayout>

如果你正在使用ViewBinding,请确保你引用了正确的绑定!

当我试图从一个活动中膨胀一个自定义对话框时,我有这个问题:

AlertDialog.Builder builder = new AlertDialog.Builder(this);

final AlertBinding alertBinding = AlertBinding.inflate(LayoutInflater.from(this), null, false);

builder.setView(binding.getRoot()); // <--- I was using binding (which is my Activity's binding), instead of alertBinding.

这就是我如何做我的自定义对话框

AlertDialog.Builder builder = new AlertDialog.Builder(MainActivity.this);
android.view.View views = getLayoutInflater().inflate(layout_file, null, false);

builder.setView(views);
dialog = builder.create();
dialog.show();

我把它改成了这个,它对我有用,我希望这对我有帮助

Dialog dialog = new Dialog(MainActivity.this);
dialog.setContentView(layout_file);
dialog.show();

如果你正在使用MaterialAlertDialog,这对我来说是有效的:

(yourChildView.parent as? ViewGroup)?.removeView(yourChildView)