[android] notifyDataSetChanged 후 Android ListView가 새로 고쳐지지 않음
내 ListFragment 코드
public class ItemFragment extends ListFragment {
private DatabaseHandler dbHelper;
private static final String TITLE = "Items";
private static final String LOG_TAG = "debugger";
private ItemAdapter adapter;
private List<Item> items;
@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
View view = inflater.inflate(R.layout.item_fragment_list, container, false);
return view;
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.setHasOptionsMenu(true);
super.onCreate(savedInstanceState);
getActivity().setTitle(TITLE);
dbHelper = new DatabaseHandler(getActivity());
items = dbHelper.getItems();
adapter = new ItemAdapter(getActivity().getApplicationContext(), items);
this.setListAdapter(adapter);
}
@Override
public void onResume() {
super.onResume();
items.clear();
items = dbHelper.getItems(); //reload the items from database
adapter.notifyDataSetChanged();
}
@Override
public void onListItemClick(ListView l, View v, int position, long id) {
super.onListItemClick(l, v, position, id);
if(dbHelper != null) { //item is edited
Item item = (Item) this.getListAdapter().getItem(position);
Intent intent = new Intent(getActivity(), AddItemActivity.class);
intent.putExtra(IntentConstants.ITEM, item);
startActivity(intent);
}
}
}
내 ListView
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" >
<ListView
android:id="@android:id/list"
android:layout_width="fill_parent"
android:layout_height="wrap_content" />
</LinearLayout>
그러나 이것은 ListView
. 앱을 다시 시작한 후에도 업데이트 된 항목이 표시되지 않습니다. 내 ItemAdapter
연장BaseAdapter
public class ItemAdapter extends BaseAdapter{
private LayoutInflater inflater;
private List<Item> items;
private Context context;
public ProjectListItemAdapter(Context context, List<Item> items) {
super();
inflater = LayoutInflater.from(context);
this.context = context;
this.items = items;
}
@Override
public int getCount() {
return items.size();
}
@Override
public Object getItem(int position) {
return items.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ItemViewHolder holder = null;
if(convertView == null) {
holder = new ItemViewHolder();
convertView = inflater.inflate(R.layout.list_item, parent,false);
holder.itemName = (TextView) convertView.findViewById(R.id.topText);
holder.itemLocation = (TextView) convertView.findViewById(R.id.bottomText);
convertView.setTag(holder);
} else {
holder = (ItemViewHolder) convertView.getTag();
}
holder.itemName.setText("Name: " + items.get(position).getName());
holder.itemLocation.setText("Location: " + items.get(position).getLocation());
if(position % 2 == 0) {
convertView.setBackgroundColor(context.getResources().getColor(R.color.evenRowColor));
} else {
convertView.setBackgroundColor(context.getResources().getColor(R.color.oddRowColor));
}
return convertView;
}
private static class ItemViewHolder {
TextView itemName;
TextView itemLocation;
}
}
누군가 제발 도와 줄 수 있습니까?
답변
다음에서 onResume
방법을 살펴보십시오 ItemFragment
.
@Override
public void onResume() {
super.onResume();
items.clear();
items = dbHelper.getItems(); // reload the items from database
adapter.notifyDataSetChanged();
}
호출하기 전에 방금 업데이트 한 notifyDataSetChanged()
것은 어댑터의 필드가 private List<Item> items;
아니라 조각의 동일하게 선언 된 필드입니다. 어댑터는 어댑터를 만들 때 전달한 항목 목록에 대한 참조를 계속 저장합니다 (예 : 조각의 onCreate). (변경 횟수 측면에서) 가장 짧지 만 코드가 예상대로 작동하도록하는 우아한 방법은 단순히 줄을 바꾸는 것입니다.
items = dbHelper.getItems(); // reload the items from database
와
items.addAll(dbHelper.getItems()); // reload the items from database
보다 우아한 솔루션 :
1) 제거 항목 private List<Item> items;
에서 ItemFragment
우리는 어댑터에 그들에게 참조를 유지해야합니다 –
2) onCreate를 다음으로 변경하십시오.
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
super.setHasOptionsMenu(true);
getActivity().setTitle(TITLE);
dbHelper = new DatabaseHandler(getActivity());
adapter = new ItemAdapter(getActivity(), dbHelper.getItems());
setListAdapter(adapter);
}
3) ItemAdapter에 메소드 추가 :
public void swapItems(List<Item> items) {
this.items = items;
notifyDataSetChanged();
}
4) onResume을 다음과 같이 변경하십시오.
@Override
public void onResume() {
super.onResume();
adapter.swapItems(dbHelper.getItems());
}
답변
다시로드 된 항목을의 전역 변수 항목에 할당 onResume()
하고 있지만 ItemAdapter
‘items’라는 자체 인스턴스 변수가 있으므로 클래스에 반영되지 않습니다 .
refreshing의 ListView
경우 ItemAdapter
목록 데이터, 즉 항목을 허용하는 클래스에 refresh ()를 추가하십시오.
class ItemAdapter
{
.....
public void refresh(List<Item> items)
{
this.items = items;
notifyDataSetChanged();
}
}
onResume()
다음 코드로 업데이트
@Override
public void onResume()
{
super.onResume();
items.clear();
items = dbHelper.getItems(); //reload the items from database
**adapter.refresh(items);**
}
답변
onResume ()에서이 줄을 변경하십시오.
items = dbHelper.getItems(); //reload the items from database
에
items.addAll(dbHelper.getItems()); //reload the items from database
문제는 새 항목 목록에 대해 어댑터에 절대 알려주지 않는다는 것입니다. (당신이 안 보인다) 당신이 당신의 어댑터에 새 목록을 통과하지 않는 경우, 바로 사용 items.addAll
하여 후 clear()
. 이렇게하면 어댑터가 참조하는 것과 동일한 목록을 수정할 수 있습니다.
답변
어댑터가 이미 설정된 경우 다시 설정해도 목록보기가 새로 고쳐지지 않습니다. 대신 먼저 목록보기에 어댑터가 있는지 확인한 다음 적절한 메서드를 호출하십시오.
목록보기를 설정하는 동안 어댑터의 새 인스턴스를 만드는 것은별로 좋은 생각이 아니라고 생각합니다. 대신 개체를 만듭니다.
BuildingAdapter adapter = new BuildingAdapter(context);
if(getListView().getAdapter() == null){ //Adapter not set yet.
setListAdapter(adapter);
}
else{ //Already has an adapter
adapter.notifyDataSetChanged();
}
또한 UI 스레드에서 새로 고침 목록을 실행하려고 할 수 있습니다.
activity.runOnUiThread(new Runnable() {
public void run() {
//do your modifications here
// for example
adapter.add(new Object());
adapter.notifyDataSetChanged()
}
});
답변
당신이 당신의 목록보기에 당신은에 그렇게하려면 중요하지 않습니다 업데이트 할 경우 onResume()
, onCreate()
또는 다른 기능에, 당신이 실현해야한다는 첫 번째 것은 당신이 어댑터의 새로운 인스턴스를 만들 필요가 없습니다 것입니다, 단지 채우기 다시 데이터와 함께 배열. 아이디어는 다음과 유사합니다.
private ArrayList<String> titles;
private MyListAdapter adapter;
private ListView myListView;
@Override
public void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
setContentView(R.layout.main_activity);
myListView = (ListView) findViewById(R.id.my_list);
titles = new ArrayList<String>()
for(int i =0; i<20;i++){
titles.add("Title "+i);
}
adapter = new MyListAdapter(this, titles);
myListView.setAdapter(adapter);
}
@Override
public void onResume(){
super.onResume();
// first clear the items and populate the new items
titles.clear();
for(int i =0; i<20;i++){
titles.add("New Title "+i);
}
adapter.notifySetDataChanged();
}
따라서 그 대답 List<Item>
에 따라 Fragment
. 첫 번째 어댑터 초기화에서 목록을 항목으로 채우고 어댑터를 목록보기로 설정합니다. 그 후에 항목을 변경할 때마다 기본에서 값을 지우고 List<Item> items
새 항목으로 다시 채우고을 호출해야 notifySetDataChanged();
합니다.
그것이 작동하는 방법입니다 :).
답변
AlexGo의 답변이 나를 위해 트릭을 수행했습니다.
getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
messages.add(m);
adapter.notifyDataSetChanged();
getListView().setSelection(messages.size()-1);
}
});
목록 업데이트는 GUI 이벤트에서 업데이트가 트리거되어 UI 스레드에 있기 전에 저에게 효과적이었습니다.
그러나 다른 이벤트 / 스레드 (예 : 앱 외부의 호출)에서 목록을 업데이트하면 업데이트가 UI 스레드에 있지 않고 getListView에 대한 호출을 무시했습니다. 위와 같이 runOnUiThread로 업데이트를 호출하면 트릭이 생겼습니다. 감사!!
답변
이 시도
@Override
public void onResume() {
super.onResume();
items.clear();
items = dbHelper.getItems(); //reload the items from database
adapter = new ItemAdapter(getActivity(), items);//reload the items from database
adapter.notifyDataSetChanged();
}