在 notifyDataSetChanged 后 Android ListView 不刷新

Posted

技术标签:

【中文标题】在 notifyDataSetChanged 后 Android ListView 不刷新【英文标题】:Android ListView not refreshing after notifyDataSetChanged 【发布时间】:2013-01-08 07:22:18 【问题描述】:

我的 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);
        
    

我的列表视图

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_
    android:layout_
    android:orientation="vertical" >

    <ListView
        android:id="@android:id/list"
        android:layout_
        android:layout_ />

</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;
    

有人可以帮忙吗?

【问题讨论】:

你测试过数据库操作是否正常吗?适配器的外观如何?此外,如果您为 adapter 引用创建一个 on 对象,为什么要在下面一行测试它是否为 null? 代码不会抛出异常,我使用调试进行了检查。所有方法都执行没有错误。是的,这是一个愚蠢的错误。 【参考方案1】:

查看ItemFragment 中的onResume 方法:

@Override
public void onResume() 
    super.onResume();
    items.clear();
    items = dbHelper.getItems(); // reload the items from database
    adapter.notifyDataSetChanged();

您在调用notifyDataSetChanged() 之前刚刚更新的不是适配器的字段private List&lt;Item&gt; items;,而是片段的相同声明字段。适配器仍然存储对您在创建适配器时传递的项目列表的引用(例如,在片段的 onCreate 中)。 使代码按预期运行的最短(就更改数量而言)但不是优雅的方法是简单地替换该行:

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

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

更优雅的解决方案:

1) 从ItemFragment 中删除项目private List&lt;Item&gt; items; - 我们只需要在适配器中保留对它们的引用

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()的事情。但无论如何感谢您的回答:) 为什么必须 clear() 并再次添加项目?这不正是notifyDataSetChanged()的目的吗? @tomsaz 你能帮我解决这个问题吗***.com/questions/28148618/… 谢谢@tomsaz Gawel,你的swapItems真的帮了我很多,我不知道为什么我的adapter.notifydatasetchanged不起作用,因为我传递的“列表”也更新了,即使我已经通过打印检查了它日志,你能解释一下这个概念吗 这个答案是正确的。问题是 ADAPTER 的 Item ArrayList 没有被更新。这意味着您可以调用 notifydatasetchanged 直到您的脸变成蓝色而没有任何效果。适配器正在使用相同的数据集更新您的数据集,因此没有任何更改。此答案中发布的解决方案的另一种替代方案可能更简洁:adapter.items = items; adapter.notifyDataSetChanged();【参考方案2】:

您将重新加载的项目分配给 onResume() 中的全局变量项目,但这不会反映在 ItemAdapter 类中,因为它有自己的实例变量,称为“项目”。

为了刷新ListView,在ItemAdapter 类中添加一个刷新(),它接受列表数据,即项目

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);**

【讨论】:

这完全正确。适配器的构造函数期望传递项目,但他只更新外部类的字段。 嗨桑托什。能不能看看类似的问题:***.com/questions/35850715/…【参考方案3】:

在 onResume() 中改变这一行

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

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

问题是您从来没有告诉您的适配器有关新项目列表的信息。如果您不想将新列表传递给您的适配器(您似乎不想),那么只需在您的clear() 之后使用items.addAll。这将确保您正在修改适配器引用的相同列表。

【讨论】:

令人困惑的是adapter.clear() 不会强制适配器意识到视图应该刷新,但adapter.add()adapter.addAll() 会。不过感谢您的回答! 请注意,我使用的是items.addAll(),而不是adapter.addAll()。唯一让适配器对更改做出反应的是notifyDataSetChanged。适配器看到变化的原因是 items 列表与适配器正在使用的列表相同。【参考方案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 的原因是我想为行着色并且列表视图显示多个数据项。我已经发布了适配器的代码。 您需要使用“this”将填充列表的代码放入我的示例代码中。而不是“活动” 在某些情况下,当您在不同的线程中运行 notifyDataSetChanged() 时它不会更新,因此上述解决方案适用于某些情况。【参考方案5】:

如果你想更新你的列表视图,如果你想在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();

因此,根据该答案,您应该在 Fragment 中使用相同的 List&lt;Item&gt;。在您的第一个适配器初始化中,您使用项目填充列表并将适配器设置为您的列表视图。之后,在您的项目的每次更改中,您都必须清除主 List&lt;Item&gt; items 中的值,然后用您的新项目再次填充它并调用 notifySetDataChanged();

这就是它的工作原理:)。

【讨论】:

感谢您的回复。我做了你提到的改变。我已经发布了我的代码。它仍然不起作用。现在添加新项目时它甚至不显示列表视图。 我已更改代码。奇怪的是,该项目没有在数据库中更新 此线程用于数据库***.com/questions/14555332/…【参考方案6】:

AlexGo 的回答帮了我的忙:

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

在从 GUI 事件触发更新之前,列表更新为我工作,因此在 UI 线程中。

但是,当我从另一个事件/线程(即来自应用程序外部的调用)更新列表时,更新不会在 UI 线程中,并且它忽略了对 getListView 的调用。像上面那样用 runOnUiThread 调用更新对我有用。谢谢!!

【讨论】:

【参考方案7】:

试试这个

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

【讨论】:

【参考方案8】:
adpter.notifyDataSetInvalidated();

在 Activity 类的onPause() 方法中试试这个。

【讨论】:

【参考方案9】:
adapter.setNotifyDataChanged()

应该可以解决问题。

【讨论】:

这里的问题应该放在哪里??【参考方案10】:

如果您的列表包含在适配器本身中,则调用更新列表的函数也应该调用notifyDataSetChanged()

从 UI 线程运行这个函数对我有用:

适配器内部的refresh() 函数

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

然后依次从 UI Thread 运行这个函数

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

【讨论】:

这确实对我产生了影响,因为更新是通过不同的线程通过网络进行的。【参考方案11】:

试试这样:

this.notifyDataSetChanged();

代替:

adapter.notifyDataSetChanged();

你必须 notifyDataSetChanged()ListView 而不是适配器类。

【讨论】:

当然不会,只有通过列表视图扩展活动的机会

以上是关于在 notifyDataSetChanged 后 Android ListView 不刷新的主要内容,如果未能解决你的问题,请参考以下文章

调用 notifydatasetchanged 后 Recyclerview 滚动到顶部

RecyclerView 在 NotifyDataSetChanged() 之后没有更新,打开和关闭屏幕后显示内容

notifyDataSetChanged 后如何避免闪烁效果

在 RecyclerView 的第一个索引处添加项目并调用 notifydatasetchange 后如何保持在滚动位置

尽管我使用了 Adapter.notifyDataSetChanged(),但数据集更改后 Android ListView 不刷新

使用数据作为 Map 的 notifyDataSetChanged 后 Android ListView 不刷新