notifyDataSetChanged 후 Android ListView가 새로 고쳐지지 않음


116

내 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;
    }
}

누군가 제발 도와 줄 수 있습니까?


2
데이터베이스 작업이 제대로 작동하는지 테스트 했습니까? 어댑터는 어떻게 생겼습니까? 또한 adapter참조 에 대한 on 객체를 만드는 경우 왜 아래에서 null 한 줄에 대해 테스트합니까?
Luksprog 2013 년

코드는 예외를 발생시키지 않으며 디버그를 사용하여 확인했습니다. 모든 메소드가 오류없이 실행됩니다. 네 그건 어리석은 실수입니다.
Coder

답변:


229

다음에서 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());
}

dbHelper 전체를 어댑터로 옮기는 것이 더 깨끗하지 않습니까? 그래서 당신은 전화 adapter.swapItems();하고 어댑터가 dbHelper.getItems()물건을 할 것입니다. 그러나 어쨌든 답변 주셔서 감사합니다 :)
Ansgar

7
clear () 및 항목을 다시 추가해야하는 이유는 무엇입니까? 그게 목적이지 notifyDataSetChanged()않나요?
Phil Ryan

1
@tomsaz이 나를 도울 수 stackoverflow.com/questions/28148618/...

1
감사합니다 @tomsaz Gawel, 귀하의 swapItems가 저에게 많은 도움이됩니다. 제가 통과하는 "목록"도 업데이트되기 때문에 내 adapter.notifydatasetchanged가 작동하지 않는 이유를 모르겠습니다. 심지어 인쇄 로그로 확인했습니다. 설명해 주시겠습니까? 개념
Kimmi Dhingra

1
정답입니다. 문제는 ADAPTER 'S Item ArrayList가 업데이트되지 않았다는 것입니다. 즉, 아무 효과없이 얼굴이 파란색이 될 때까지 notifydatasetchanged를 호출 할 수 있습니다. 어댑터가 동일한 데이터 세트로 데이터 세트를 업데이트하므로 변경 사항이 없습니다. 이 답변에 게시 된 솔루션에 대한 또 다른 대안은 다음과 같습니다. adapter.items = items; adapter.notifyDataSetChanged ();
Ray Li

23

다시로드 된 항목을의 전역 변수 항목에 할당 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);**
}

1
이것은 정확히 맞습니다. 어댑터의 생성자는 항목이 전달 될 것으로 예상하지만 외부 클래스의 필드 만 업데이트합니다.
LuxuryMode '2013

안녕 Santhosh. 당신은 비슷한 문제를 살펴 수 : stackoverflow.com/questions/35850715/...

8

onResume ()에서이 줄을 변경하십시오.

items = dbHelper.getItems(); //reload the items from database

items.addAll(dbHelper.getItems()); //reload the items from database

문제는 새 항목 목록에 대해 어댑터에 절대 알려주지 않는다는 것입니다. (당신이 안 보인다) 당신이 당신의 어댑터에 새 목록을 통과하지 않는 경우, 바로 사용 items.addAll하여 후 clear(). 이렇게하면 어댑터가 참조하는 것과 동일한 목록을 수정할 수 있습니다.


그 혼란 adapter.clear()보기가 새로 고침해야 실현하기 위해 어댑터를 강제하지 않지만, adapter.add()또는 adapter.addAll()한다. 그래도 답변 주셔서 감사합니다!
w3bshark

내가 사용 items.addAll()하고 있었고 adapter.addAll ()이 아니라는 점에 유의하십시오 . 어댑터가 변경 사항에 반응하도록하는 유일한 방법은 notifyDataSetChanged. 어댑터가 변경 사항을 확인하는 이유는 items목록이 어댑터가 사용중인 목록과 동일하기 때문입니다.
Justin Breitfeller 2015-08-13

4

어댑터가 이미 설정된 경우 다시 설정해도 목록보기가 새로 고쳐지지 않습니다. 대신 먼저 목록보기에 어댑터가 있는지 확인한 다음 적절한 메서드를 호출하십시오.

목록보기를 설정하는 동안 어댑터의 새 인스턴스를 만드는 것은별로 좋은 생각이 아니라고 생각합니다. 대신 개체를 만듭니다.

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()  
        }
});

UI 스레드를 구현하는 방법을 잘 모르겠습니다. 내 주요 활동에는 3 개의 조각 (탭)이 있으며 질문의 코드는 목록보기를 포함하는 조각 중 하나와 관련이 있습니다. 항목을 전달하는 이유 ItemAdapter는 행에 색상을 지정하고 목록보기에 여러 데이터 항목이 표시 되기 때문입니다 . 어댑터 코드를 게시했습니다.
Coder

"this"를 사용하여 내 예제 코드에 목록을 채우는 코드를 입력해야합니다. "활동"대신
AlexGo 2013-01-28

어떤 경우에는 다른 스레드에서 notifyDataSetChanged ()를 실행할 때 업데이트되지 않으므로 위의 솔루션이 경우에 따라 적합합니다.
Ayman Al-Absi 2014

4

당신이 당신의 목록보기에 당신은에 그렇게하려면 중요하지 않습니다 업데이트 할 경우 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();합니다.

그것이 작동하는 방법입니다 :).


답장을 보내 주셔서 감사합니다. 말씀하신대로 변경했습니다. 내 코드를 게시했습니다. 여전히 작동하지 않습니다. 이제 새 항목이 추가 될 때 목록보기도 표시되지 않습니다.
Coder

코드를 변경했습니다. 관찰해야 할 이상한 점은 항목이 DB에서 업데이트되지 않는다는 것입니다.
코더

이 스레드는 데이터베이스입니다 stackoverflow.com/questions/14555332/...
코더

3

AlexGo의 답변이 나를 위해 트릭을 수행했습니다.

getActivity().runOnUiThread(new Runnable() {
        @Override
        public void run() {
         messages.add(m);
         adapter.notifyDataSetChanged();
         getListView().setSelection(messages.size()-1);
        }
});

목록 업데이트는 GUI 이벤트에서 업데이트가 트리거되어 UI 스레드에 있기 전에 저에게 효과적이었습니다.

그러나 다른 이벤트 / 스레드 (예 : 앱 외부의 호출)에서 목록을 업데이트하면 업데이트가 UI 스레드에 있지 않고 getListView에 대한 호출을 무시했습니다. 위와 같이 runOnUiThread로 업데이트를 호출하면 트릭이 생겼습니다. 감사!!


3

이 시도

@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();
}

3
adpter.notifyDataSetInvalidated();

onPause()Activity 클래스의 방법으로 이것을 시도하십시오 .


1
adapter.setNotifyDataChanged()

트릭을해야합니다.


3
여기에 질문을 어디에 넣을까요 ??
swiftBoy

1

목록이 Adapter 자체에 포함되어있는 경우 목록을 업데이트하는 함수를 호출하면 notifyDataSetChanged().

UI 스레드에서이 함수를 실행하면 나에게 트릭이 생겼습니다.

refresh()어댑터 내부 의 기능

public void refresh(){
    //manipulate list
    notifyDataSetChanged();
}

그런 다음 UI 스레드에서이 함수를 실행합니다.

getActivity().runOnUiThread(new Runnable() { 
    @Override
    public void run() {
          adapter.refresh()  
    }
});

업데이트가 다른 스레드를 통해 네트워크를 통해 이루어 졌기 때문에 이것은 실제로 저에게 차이를 만들었습니다.

0

다음과 같이 시도하십시오.

this.notifyDataSetChanged();

대신에:

adapter.notifyDataSetChanged();

어댑터 클래스 notifyDataSetChanged()ListView아니 어야합니다 .


물론 그렇지 않습니다. 활동이 목록보기에 의해 확장되는 유일한 기회입니다.
cmario
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.