Почему мне нужно вызвать removeView(), чтобы добавить View в мой LinearLayout

Я получаю эту ошибку, но не знаю точно, почему:

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

Часть, в которую я добавляю View - это строка, на которую указывает ошибка. Я предоставляю свой код адаптера, чтобы вы, ребята, могли лучше понять, что я делаю и почему я получаю эту ошибку. Пожалуйста, дайте мне знать, если потребуется дополнительная информация. Заранее спасибо.

адаптер

private class InnerAdapter extends BaseAdapter{
    String[] array = new String[] {"12\nAM","1\nAM", "2\nAM", "3\nAM", "4\nAM", "5\nAM", 
                                "6\nAM", "7\nAM", "8\nAM", "9\nAM", "10\nAM", "11\nAM",
                                "12\nPM", "1\nPM", "2\nPM", "3\nPM", "4\nPM", "5\nPM",
                                "6\nPM", "7\nPM", "8\nPM", "9\nPM", "10\nPM", "11\nPM"};
    TextView[] views = new TextView[24];

    public InnerAdapter() {
        TextView create = new TextView(DayViewActivity.this);
        LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(0, (int)TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 62, getResources().getDisplayMetrics()), 1.0f);
        params.topMargin = (int)TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 1, getResources().getDisplayMetrics());
        params.bottomMargin = (int)TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, 1, getResources().getDisplayMetrics());
        create.setLayoutParams(params);
        create.setBackgroundColor(Color.BLUE);
        create.setText("Test");
        views[0] = create;
        for(int i = 1; i < views.length; i++) {
            views[i] = null;
        }
    }

    @Override
    public int getCount() {
        // TODO Auto-generated method stub
        return array.length;
    }

    @Override
    public Object getItem(int position) {
        // TODO Auto-generated method stub
        return null;
    }

    @Override
    public long getItemId(int position) {
        // TODO Auto-generated method stub
        return position;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if(convertView == null) {
            LayoutInflater inflater = LayoutInflater.from(parent.getContext());
            convertView = inflater.inflate(R.layout.day_view_item, parent, false);
        }

        ((TextView)convertView.findViewById(R.id.day_hour_side)).setText(array[position]);
        LinearLayout layout = (LinearLayout)convertView.findViewById(R.id.day_event_layout);
        if(views[position] != null) {
            layout.addView((TextView)views[position], position);
        }

        return convertView;
    }

}

XML

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="61dp"
    android:orientation="horizontal" >



    <LinearLayout 
        android:layout_width="wrap_content"
        android:layout_height="61dp"
        android:orientation="vertical">
        <TextView 
            android:id="@+id/day_hour_side"
            android:layout_width="wrap_content"
            android:layout_height="60dp"
            android:text="12AM"
            android:background="#bebebe"
            android:layout_weight="0"
            android:textSize="10dp"
            android:paddingLeft="5dp"
            android:paddingRight="5dp"/>
        <TextView 
            android:layout_width="match_parent"
            android:layout_height="1dp"
            android:layout_weight="0"
            android:background="#000000"
            android:id="@+id/hour_side_divider"/>
    </LinearLayout>
    <LinearLayout 
        android:layout_width="0dp"
        android:layout_height="61dp"
        android:orientation="vertical"
        android:layout_weight="1">
        <LinearLayout 
            android:id="@+id/day_event_layout"
            android:layout_width="match_parent"
            android:layout_height="60dp"
            android:orientation="horizontal" ></LinearLayout>
        <TextView 
            android:layout_width="match_parent"
            android:layout_height="1dp"
            android:background="#000000"
            android:id="@+id/event_side_divider" />
    </LinearLayout>


</LinearLayout>

2 ответа

Решение

Вы не сказали, когда вы получаете это исключение (когда приложение запускается или когда вы прокручиваете GridView вверх и вниз) но это нормально. views массив имеет одно значение, которое не null(первая запись в этом массиве установлена ​​в TextView что вы создаете), и, скорее всего, вы будете пытаться повторно добавить это TextView в какой-то момент. Также родитель AdapterView может позвонить getView метод несколько раз, чтобы заставить детей измерить.

В любом случае, вы точно не знаете, что пытаетесь сделать, но текущий подход неверен.

Сначала вы создаете массив с одним TextView а остальные значения установлены в null и вы в основном ничего с этим не делаете (но, возможно, это не полный код?!). Во-вторых, вы не должны хранить массив Views особенно в детстве AdapterView(лайк GridView, ListView и т.д.) который имеет механизм для переработки своих детей. В-третьих, вы не приняли во внимание механизм утилизации GridView, Например, вы добавляете TextView для первого элемента, но вы не отменяете эти изменения в getView так что если этот первый ряд View(который содержит добавленный TextView) будет переработан, в результате вы получите ряд View содержащие ранее добавленные TextView в строках, где вы этого не хотите.

Я уже принял ответ, но подумал, что добавлю это, так как это поможет объяснить некоторые вещи о ListView (и по определению, GridView также), что кто-то, узнающий об этом, может понять. Я был озадачен переработкой Views в ListView, и эта статья, которую я нашел, великолепна. Объясняет это хорошо. Надеюсь, это поможет всем, кто не совсем понимает, как работают ListView и адаптеры, что было для меня очевидной проблемой.

Другие вопросы по тегам