我的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;
}
}
有人可以帮忙吗?
答案 0 :(得分:219)
查看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());
}
答案 1 :(得分:22)
您正在将重新加载的项目分配给onResume()
中的全局变量项,但这不会反映在ItemAdapter
类中,因为它有自己的实例变量名为'items'。
要刷新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);**
}
答案 2 :(得分:8)
在onResume()中更改此行
items = dbHelper.getItems(); //reload the items from database
到
items.addAll(dbHelper.getItems()); //reload the items from database
问题是你永远不会告诉你的适配器有关新项列表的信息。如果您不想将新列表传递给适配器(因为您似乎没有),那么只需在items.addAll
之后使用clear()
。这将确保您修改适配器引用的相同列表。
答案 3 :(得分:4)
如果已设置适配器,则再次进行设置将不会刷新列表视图。而是首先检查listview是否有适配器,然后调用适当的方法。
我认为在设置列表视图时创建适配器的新实例并不是一个好主意。相反,创建一个对象。
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()
}
});
答案 4 :(得分: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();
。
这就是它的工作原理:)。
答案 5 :(得分: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调用更新对我来说是个窍门。谢谢!
答案 6 :(得分: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();
}
答案 7 :(得分:2)
adpter.notifyDataSetInvalidated();
在Activity类的onPause()
方法中尝试此操作。
答案 8 :(得分:1)
adapter.setNotifyDataChanged()
应该这样做。
答案 9 :(得分:0)
试试这样:
this.notifyDataSetChanged();
而不是:
adapter.notifyDataSetChanged();
您必须notifyDataSetChanged()
到ListView
而不是适配器类。
答案 10 :(得分:0)
如果您的列表包含在适配器本身中,则调用更新列表的函数也应调用notifyDataSetChanged()
。
从UI线程运行此功能对我来说很有用:
适配器内的refresh()
功能
public void refresh(){
//manipulate list
notifyDataSetChanged();
}
然后依次从UI Thread
运行此函数getActivity().runOnUiThread(new Runnable() {
@Override
public void run() {
adapter.refresh()
}
});