Android — добавление и отображение элементов в ListView по одному с помощью ArrayAdapter

Я использую ArrayAdapter для добавления элементов в пользовательский ListView и отображения результатов в своем приложении для Android. Проблема, с которой я сталкиваюсь, заключается в том, что ArrayAdapter, кажется, ждет, пока все элементы не будут в нем, прежде чем он покажет представление. То есть, когда я добавляю элементы в ArrayAdapter и вызываю notifyDataSetChanged, он не обновляет ListView для отображения добавленного элемента. Он ожидает, пока все элементы не будут добавлены, и вызов GetView перед отображением элементов.

Я хотел бы, чтобы он отображал элемент сразу после его добавления в ListView. Это возможно?

Я считаю, что соответствующий код следующий:

r_adapter = new ReminderAdapater(Activity_ContentSearch.this, R.layout.search_listitem, myList);
listView.setAdapter(r_adapter);
...
r_adapter.notifyDataSetChanged();
r_adapter.clear();
for(int i = 0; i < myList.size(); i++)
{
    r_adapter.add(myList.get(i));
    r_adapter.notifyDataSetChanged();
}

Как видите, несмотря на то, что я вызываю notifyDataSetChanged после метода добавления, он фактически не обновляет представление. После завершения вышеуказанного цикла представление окончательно обновляется (исходя из того, что я знаю, это потому, что GetView не вызывается до тех пор, пока не будет выполнен этот раздел кода).

Я безуспешно пытался переопределить метод добавления моего пользовательского ArrayAdapter, поскольку у меня нет доступа к представлению в этом методе.

Любая помощь приветствуется :)

Бара


person Bara    schedule 25.01.2010    source источник


Ответы (1)


Пользовательский интерфейс Android является однопоточным. Вы не возвращаете управление Android из основного потока приложения каждый раз, когда добавляете запись в адаптер. Следовательно, у Android не будет возможности отобразить ваши записи, пока вы не вернете управление, и вы не сделаете этого, пока полностью не заполните свой адаптер.

Вот пример, показывающий использование AsyncTask для заполнения ArrayAdapter постепенно через фоновый поток.

/***
  Copyright (c) 2008-2012 CommonsWare, LLC
  Licensed under the Apache License, Version 2.0 (the "License"); you may not
  use this file except in compliance with the License. You may obtain   a copy
  of the License at http://www.apache.org/licenses/LICENSE-2.0. Unless required
  by applicable law or agreed to in writing, software distributed under the
  License is distributed on an "AS IS" BASIS,   WITHOUT WARRANTIES OR CONDITIONS
  OF ANY KIND, either express or implied. See the License for the specific
  language governing permissions and limitations under the License.

  From _The Busy Coder's Guide to Android Development_
    http://commonsware.com/Android
*/

package com.commonsware.android.async;

import android.app.ListActivity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.os.SystemClock;
import android.widget.ArrayAdapter;
import android.widget.Toast;
import java.util.ArrayList;

public class AsyncDemo extends ListActivity {
  private static final String[] items={"lorem", "ipsum", "dolor",
                                      "sit", "amet", "consectetuer",
                                      "adipiscing", "elit", "morbi",
                                      "vel", "ligula", "vitae",
                                      "arcu", "aliquet", "mollis",
                                      "etiam", "vel", "erat",
                                      "placerat", "ante",
                                      "porttitor", "sodales",
                                      "pellentesque", "augue",
                                      "purus"};
  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);

    setListAdapter(new ArrayAdapter<String>(this,
                        android.R.layout.simple_list_item_1,
                        new ArrayList<String>()));

    new AddStringTask().execute();
  }

  class AddStringTask extends AsyncTask<Void, String, Void> {
    @Override
    protected Void doInBackground(Void... unused) {
      for (String item : items) {
        publishProgress(item);
        SystemClock.sleep(200);
      }

      return(null);
    }

    @SuppressWarnings("unchecked")
    @Override
    protected void onProgressUpdate(String... item) {
      ((ArrayAdapter<String>)getListAdapter()).add(item[0]);
    }

    @Override
    protected void onPostExecute(Void unused) {
      Toast
        .makeText(AsyncDemo.this, "Done!", Toast.LENGTH_SHORT)
        .show();
    }
  }
}
person CommonsWare    schedule 25.01.2010