ArrayIndexOutOfBoundsException с пользовательским адаптером Android для нескольких представлений в ListView

185

Я пытаюсь создать собственный Адаптер для моего ListView, поскольку каждый элемент в списке может иметь различное представление (ссылка, переключатель или радиогруппа), но когда я пытаюсь запустить Activity, использующую ListView, я получаю сообщение об ошибке и приложение останавливается. Приложение предназначено для платформы Android 1.6.

Код:

public class MenuListAdapter extends BaseAdapter {
 private static final String LOG_KEY = MenuListAdapter.class.getSimpleName();

 protected List<MenuItem> list;
 protected Context ctx;
 protected LayoutInflater inflater;

 public MenuListAdapter(Context context, List<MenuItem> objects) {
  this.list = objects;
  this.ctx = context;
  this.inflater = (LayoutInflater)this.ctx.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
 }

 @Override
 public View getView(int position, View convertView, ViewGroup parent) {
  Log.i(LOG_KEY, "Position: " + position + "; convertView = " + convertView + "; parent=" + parent);
  MenuItem item = list.get(position);
  Log.i(LOG_KEY, "Item=" + item );

        if (convertView == null)  {
            convertView = this.inflater.inflate(item.getLayout(), null);
        }

        return convertView;
 }

 @Override
 public boolean areAllItemsEnabled() {
  return false;
 }

 @Override
 public boolean isEnabled(int position) {
  return true;
 }

 @Override
 public int getCount() {
  return this.list.size();
 }

 @Override
 public MenuItem getItem(int position) {
  return this.list.get(position);
 }

 @Override
 public long getItemId(int position) {
  return position;
 }

 @Override
 public int getItemViewType(int position) {
  Log.i(LOG_KEY, "getItemViewType: " + this.list.get(position).getLayout());
  return this.list.get(position).getLayout();
 }

 @Override
 public int getViewTypeCount() {
  Log.i(LOG_KEY, "getViewTypeCount: " + this.list.size());
  return this.list.size();
 }

}

Я получаю ошибку:

    java.lang.ArrayIndexOutOfBoundsException
  at android.widget.AbsListView$RecycleBin.addScrapView(AbsListView.java:3523)
  at android.widget.ListView.measureHeightOfChildren(ListView.java:1158)
  at android.widget.ListView.onMeasure(ListView.java:1060)
  at android.view.View.measure(View.java:7703)

Я знаю, что приложение возвращается из, getViewи все кажется в порядке.

Любые идеи о том, что может быть причиной этого, будут оценены.

Спасибо,

Дан

Дэн
источник
Привет, Дэн, ты уверен, что в getCount есть все элементы, которые будут отображаться в твоем ListView ??? public int getCount () {return this.list.size (); }
Jorgesys
4
Они не могли придумать менее описательное сообщение об ошибке ...
JoeyJ

Ответы:

515

Тип представления товара, с которого вы возвращаетесь

getItemViewType()есть >= getViewTypeCount().

Ромэн Гай
источник
24
Этот противный. Нет способа отладить это. Большое спасибо, Romain Guy за объяснение.
Яр
113
Другое дело, ViewType должен начинаться с 0.
HelmiB
16
Вы получили 38 голосов за ответ до сих пор, так что это может быть идеальный ответ, но, пожалуйста, добавьте еще немного текста для объяснения проблемы и как ее решить .. потому что я до сих пор не могу многое понять из этого.
Чираг Патель
15
Для тех, кто озадачен, ViewType должен возвращать нулевой индекс. например (0, 1, 2 ...). в моем случае я попытался повторно использовать R.layout ... для ViewType. Я думаю, внутренне значение ViewType используется в качестве индекса для рециркуляции пула, а не в качестве ключа.
Самуил
2
Спасибо, что объяснили эту ошибку, в отличие от разработчиков Android, которые, как обычно, не удосужились упомянуть этот жизненно важный недостаток в своей документации.
WKS
17

Принятый ответ правильный. Вот что я делаю, чтобы избежать проблемы:

public enum FoodRowType {
    ONLY_ELEM,
    FIRST_ELEM,
    MID_ELEM,
    LAST_ELEM
}

@Override
public int getViewTypeCount() {
    return FoodRowType.values().length;
}

@Override
public int getItemViewType(int position) {
    return rows.get(position).getViewType();  //returns one of the above types
}
mtbomb
источник
Может быть хорошо, только если getItemViewTypeне вернуть int
dVaffection
0

Проблема возникает, когда значение getItemType неверно. Это значение должно быть целым числом и должно быть от 0 до getViewTypeCount () - 1.

Сунил Джайн
источник
-3

Причина, скорее всего, в том, что метод getItemViewType возвращает неправильные значения! Каждая строка в списке - это один вид. Во время прокрутки getItemViewType достигает большего, чем количество просмотров.

Что делать? Как избежать проблемы?

Сначала определите вид (строку) в вашем списке, который отображается первым. При прокрутке используйте уравнение модулятора

    @Override
    public int getItemViewType(int position) {
        return choseType(position);//function to use modular equation
    }
    @Override
    public int getViewTypeCount() {
        return 10;
    }

В этом примере есть десять видов (10 строк).

private  int choseType(int position){
    if(position%10==0)
        return 0;
    else if(position%10==1)
        return 1;
    else if(position%10==2)
        return 2;
    else if(position%10==3)
        return 3;
    else if(position%10==4)
        return 4;
    else if(position%10==5)
        return 5;
    else if(position%10==6)
        return 6;
    else if(position%10==7)
        return 7;
    else if(position%10==8)
        return 8;
    else
        return 9;


}

Важный

Некоторые пользователи упоминали об этом методе в стеке

public int getViewTypeCount () и public int getItemViewType (int position) исправляют, например Tooglebutton, автоматически включают проверку состояния true при прокрутке ... это большая ошибка. Если вам не нужно автоматическое enbale на scrool, просто сделайте

toogleButton.setChecked(false);

на метод переопределения getView.

Beyaz
источник