notifyDataSetChange从自定义适配器不起作用


125

当我重新填充我的时ListView,我从我的容器中调用一个具体的方法Adapter

问题

当我updateReceiptsList从拨打电话时Adapter,数据会刷新,但我ListView无法反映更改。

问题

为什么ListView我打电话时不显示新数据notifyDataSetChanged

适配器

public class ReceiptListAdapter extends BaseAdapter {

    public List<Receipt> receiptlist;
    private Context context;
    private LayoutInflater inflater;
    private DateHelpers dateH;

    public ReceiptListAdapter(Activity activity, Context mcontext, List<Receipt> rl) {
        context = mcontext;
        receiptlist = rl;
        Collections.reverse(receiptlist);
        inflater = (LayoutInflater)activity.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        dateH = new DateHelpers();
    }

    @Override
    public int getCount() {
        try {
            int size = receiptlist.size();
            return size;
        } catch(NullPointerException ex) {
            return 0;
        }
    }

    public void updateReceiptsList(List<Receipt> newlist) {
        receiptlist = newlist;
        this.notifyDataSetChanged();
    }

    @Override
    public Receipt getItem(int i) {
        return receiptlist.get(i);
    }

    @Override
    public long getItemId(int i) {
        return receiptlist.get(i).getReceiptId() ;
    }

    private String getPuntenString(Receipt r) {
        if(r.getPoints().equals("1")) {
            return "1 punt";
        }
        return r.getPoints()+" punten";
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        View vi=convertView;

        final Receipt receipt = receiptlist.get(position);
        ReceiptViewHolder receiptviewholder;
        Typeface tf_hn = Typeface.createFromAsset(context.getAssets(), "helveticaneue.ttf");        
        Typeface tf_hn_bold = Typeface.createFromAsset(context.getAssets(), "helveticaneuebd.ttf");

        if (vi == null) { //convertview==null
            receiptviewholder = new ReceiptViewHolder();
            vi = inflater.inflate(R.layout.view_listitem_receipt, null);
            vi.setOnClickListener(null);
            vi.setOnLongClickListener(null);
            vi.setLongClickable(false);
            receiptviewholder.shop = (TextView) vi.findViewById(R.id.tv_listitemreceipt_shop);
            receiptviewholder.date = (TextView) vi.findViewById(R.id.tv_listitemreceipt_date);
            receiptviewholder.price = (TextView) vi.findViewById(R.id.tv_listitemreceipt_price);
            receiptviewholder.points = (TextView) vi.findViewById(R.id.tv_listitemreceipt_points);
            receiptviewholder.shop.setTypeface(tf_hn_bold);
            receiptviewholder.price.setTypeface(tf_hn_bold);
            vi.setTag(receiptviewholder);
        }else{//convertview is not null
            receiptviewholder = (ReceiptViewHolder)vi.getTag();
        }

        receiptviewholder.shop.setText(receipt.getShop());
        receiptviewholder.date.setText(dateH.timestampToDateString(Long.parseLong(receipt.getPurchaseDate())));
        receiptviewholder.price.setText("€ "+receipt.getPrice());
        receiptviewholder.points.setText(getPuntenString(receipt));

        vi.setClickable(false);
        return vi;
    }

    public static class ReceiptViewHolder {
        public TextView shop;
        public TextView date;
        public TextView price;
        public TextView points;
    }

    public Object getFilter() {
        // XXX Auto-generated method stub
        return null;
    }

}

- 编辑:

找到解决方法

我现在要做一些功能代码:

listview.setAdapter( new ReceiptListAdapter(activity,mcontext, -new dataset-);

起作用,但不起作用。


stackoverflow.com/a/4198569/2382964,嗨贾斯珀(Jasper),请参阅此链接...这将为您提供帮助。
Tushar Pandey

尝试..其他的方法,如notifyItemInserted,notifyItemRemoved等
Reejesh PK

Answers:


334

从更改您的方法

public void updateReceiptsList(List<Receipt> newlist) {
    receiptlist = newlist;
    this.notifyDataSetChanged();
}

public void updateReceiptsList(List<Receipt> newlist) {
    receiptlist.clear();
    receiptlist.addAll(newlist);
    this.notifyDataSetChanged();
}

因此,您可以在适配器中保留与数据集相同的对象。


考虑使用父对象ReceiptListObject而不是List对象,那么该怎么办才能解决此问题?
prom85

@ prom85 ArrayAdapter甚至可以绑定到列表或数组以外的对象吗?我不知道。
tolgap 2014年

它是关于a的,BaseAdapter并且此适配器不知道将其绑定到哪些数据...因此,如果我有一个自定义对象并使用该对象的自定义功能(例如custObject.getCount()custObject.getChildAt(int i)例如),并且我想交换该对象,notifyDataSetChanged则无法正常工作......反正我觉得有它从来没有出现这个问题ArrayAdapter
prom85

3
您能否解释为什么第一种方法不起作用,而第二种方法对BaseAdapter起作用?
Tooroop

1
像“谢谢”或“ +1”之类的评论是不允许的,但在某些答案上,我真的很感谢:)
Muhammad Saqib

24

我有同样的问题,我意识到这一点。当我们创建适配器并将其设置为listview时,listview将指向对象所在的内存中适配器所在的某个位置,该对象中的数据将显示在listview中。

adapter = new CustomAdapter(data);
listview.setadapter(adapter);

如果我们再次使用另一个数据为适配器创建一个对象,然后notifydatasetchanged():

adapter = new CustomAdapter(anotherdata);
adapter.notifyDataSetChanged();

这将不会影响listview中的数据,因为列表指向的是不同的对象,此对象对适配器中的新对象一无所知,并且notifyDataSetChanged()不起作用。因此,我们应该更改对象中的数据,并避免再次为适配器创建新对象


10
您正在重新创建适配器对象,效率不高
Alezis

2
@Alezis我想这正是Nhan的意思。
Neeraj Sewani,

17

正如我已经解释了这个问题的原因以及如何在其他答案线程中处理它的原因 这里。我仍然在这里分享解决方案摘要。

主要原因之一 notifyDataSetChanged()不适用于您的是-

您的适配器失去对列表的引用

在创建新列表并将其添加到 Adapter。始终遵循以下准则:

  1. 初始化 arrayList全局声明时。
  2. 将列表直接添加到适配器,而无需检查空值和空值。将适配器直接设置到列表中(不要检查任何条件)。适配器向您保证无论您在哪里更改arrayList都会照顾到它,但绝不会丢失引用。
  3. 始终修改arrayList本身中的数据(如果您的数据是全新的,则可以调用adapter.clear()arrayList.clear()之前实际上将数据添加到列表中),但不设置适配器,即如果新的数据是在人口arrayList不仅仅是 adapter.notifyDataSetChanged()

希望这可以帮助。


1
谢谢!#2的提示有所帮助。
Cheruby

4

也许尝试刷新您的ListView:

receiptsListView.invalidate()

编辑:我想到另一个想法。仅作记录,请尝试禁用列表视图缓存:

<ListView
    ...
    android:scrollingCache="false"
    android:cacheColorHint="@android:color/transparent"
    ... />

很奇怪,我的最后一个想法是在数据更改后将适配器重新分配给列表视图。但是我想你也已经尝试过了。
RafalGałka2013年

3

我在使用时遇到了同样的问题 ListAdapter

我让Android Studio为我实现方法,这就是我得到的:

public class CustomAdapter implements ListAdapter {
    ...
    @Override
    public void registerDataSetObserver(DataSetObserver observer) {

    }

    @Override
    public void unregisterDataSetObserver(DataSetObserver observer) {

    }
    ...
}

问题在于这些方法不会调用super实现,因此notifyDataSetChange永远不会调用。

手动删除这些替代或添加超级调用,它应该可以再次工作。

@Override
public void registerDataSetObserver(DataSetObserver observer) {
    super.registerDataSetObserver(observer);
}

@Override
public void unregisterDataSetObserver(DataSetObserver observer) {
    super.unregisterDataSetObserver(observer);
}

2
class StudentAdapter extends BaseAdapter {
    ArrayList<LichHocDTO> studentList;

    private void capNhatDuLieu(ArrayList<LichHocDTO> list){
        this.studentList.clear();
        this.studentList.addAll(list);
        this.notifyDataSetChanged();
    }
}

你可以试试。它对我有用



1

如果您偶然碰到了这个线程并想知道为什么adapter.invaidate()adapter.clear()方法不存在,那可能是因为您可能正在使用RecyclerView.Adapter而不是BaseAdapter该问题的问询者使用的方法。如果清除list或未arraylist解决您的问题,则可能发生两个或多个实例的情况adapter为前:

主要活动

...

adapter = new CustomAdapter(list);
adapter.notifyDataSetChanged();
recyclerView.setAdapter(adapter);

...


SomeFragment

...

adapter = new CustomAdapter(newList);
adapter.notifyDataSetChanged();

...

如果在第二种情况下,您期望在回收者视图中的膨胀视图列表中发生更改,则不会发生这种情况,因为在第二次adapter创建未附加到回收者视图中的的新实例时。notifyDataSetChanged在第二个适配器中进行设置不会更改回收器视图的内容。为此,在SomeFragment创建回收器视图的新实例,并将其附加到适配器的新实例。

碎片

...

recyclerView = new RecyclerView();
adapter = new CustomAdapter();
recyclerView.setAdapter(adapter);

...

虽然,我不建议在同一适配器和回收站视图中创建多个实例。


0

添加此代码

runOnUiThread(new Runnable() { public void run() {
               adapter = new CustomAdapter(anotherdata);
            adapter.notifyDataSetChanged();
            }
        });

0

我有同样的问题,但我刚完成!!

您应该更改为

public class ReceiptListAdapter extends BaseAdapter {

    public List<Receipt> receiptlist;
    private Context context;
    private LayoutInflater inflater;
    private DateHelpers dateH;
    private List<ReceiptViewHolder> receiptviewlist;

    public ReceiptListAdapter(Activity activity, Context mcontext, List<Receipt> rl) {
        context = mcontext;
        receiptlist = rl;
        receiptviewlist = new ArrayList<>();
        receiptviewlist.clear();
        for(int i = 0; i < receiptlist.size(); i++){
          ReceiptViewHolder receiptviewholder = new ReceiptViewHolder();
          receiptviewlist.add(receiptviewholder);
        }
        Collections.reverse(receiptlist);
        inflater = (LayoutInflater)activity.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        dateH = new DateHelpers();
    }

    @Override
    public int getCount() {
        try {
            int size = receiptlist.size();
            return size;
        } catch(NullPointerException ex) {
            return 0;
        }
    }

    public void updateReceiptsList(List<Receipt> newlist) {
        receiptlist = newlist;
        this.notifyDataSetChanged();
    }

    @Override
    public Receipt getItem(int i) {
        return receiptlist.get(i);
    }

    @Override
    public long getItemId(int i) {
        return receiptlist.get(i).getReceiptId() ;
    }

    private String getPuntenString(Receipt r) {
        if(r.getPoints().equals("1")) {
            return "1 punt";
        }
        return r.getPoints()+" punten";
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        View vi=convertView;

        final Receipt receipt = receiptlist.get(position);
        ReceiptViewHolder receiptviewholder;
        Typeface tf_hn = Typeface.createFromAsset(context.getAssets(), "helveticaneue.ttf");        
        Typeface tf_hn_bold = Typeface.createFromAsset(context.getAssets(), "helveticaneuebd.ttf");

        if (vi == null) { //convertview==null
            ReceiptViewHolder receiptviewholder = receiptviewlist.get(position);
            vi = inflater.inflate(R.layout.view_listitem_receipt, null);
            vi.setOnClickListener(null);
            vi.setOnLongClickListener(null);
            vi.setLongClickable(false);
            receiptviewholder.shop = (TextView) vi.findViewById(R.id.tv_listitemreceipt_shop);
            receiptviewholder.date = (TextView) vi.findViewById(R.id.tv_listitemreceipt_date);
            receiptviewholder.price = (TextView) vi.findViewById(R.id.tv_listitemreceipt_price);
            receiptviewholder.points = (TextView) vi.findViewById(R.id.tv_listitemreceipt_points);
            receiptviewholder.shop.setTypeface(tf_hn_bold);
            receiptviewholder.price.setTypeface(tf_hn_bold);
            vi.setTag(receiptviewholder);
        }else{//convertview is not null
            receiptviewholder = (ReceiptViewHolder)vi.getTag();
        }

        receiptviewholder.shop.setText(receipt.getShop());
        receiptviewholder.date.setText(dateH.timestampToDateString(Long.parseLong(receipt.getPurchaseDate())));
        receiptviewholder.price.setText("€ "+receipt.getPrice());
        receiptviewholder.points.setText(getPuntenString(receipt));

        vi.setClickable(false);
        return vi;
    }

    public static class ReceiptViewHolder {
        public TextView shop;
        public TextView date;
        public TextView price;
        public TextView points;
    }

    public Object getFilter() {
        // XXX Auto-generated method stub
        return null;
    }

}

0

我的情况不同,但其他情况可能相同

对于仍然找不到解决方案并尝试了上述所有操作的用户,如果您在片段内部使用适配器,则说明它无法正常工作的原因 fragment could be recreating so the adapter is recreating everytime the fragment recreate

您应该在初始化之前验证适配器和对象列表是否为空

if(adapter == null){
  adapter = new CustomListAdapter(...);
}
...

if(objects == null){
  objects = new ArrayList<>();
}

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.