Android - Добавление и отображение элементов в ListView по одному с помощью ArrayAdapter
-
22-09-2019 - |
Вопрос
Я использую 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 после метода add , он фактически не обновляет представление.После завершения вышеупомянутого цикла представление окончательно обновляется (исходя из того, что я знаю, это потому, что getView не вызывается до тех пор, пока не будет выполнен этот раздел кода).
Я попытался переопределить метод add моего пользовательского ArrayAdapter, но безуспешно, поскольку у меня нет доступа к представлению в этом методе.
Любая помощь была бы желанна :)
Бара
Решение
Пользовательский интерфейс 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();
}
}
}