119

私の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_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;
    }
}

誰か助けてくれませんか?

4

11 に答える 11

235

あなたの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

よりエレガントなソリューション:

private List<Item> items;1) から項目を削除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());
}
于 2013-02-03T01:52:54.193 に答える
25

再ロードされた項目を のグローバル変数項目に割り当てていますonResume()が、これはクラスに反映されません。これItemAdapterは、'items' という独自のインスタンス変数があるためです。

リフレッシュするには、リストデータ、つまりアイテムを受け入れるクラスにListViewrefresh() を追加します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);**
}
于 2013-01-30T13:46:09.367 に答える
8

onResume() で、この行を変更します

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

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

問題は、アダプターに新しい項目リストについて決して伝えていないことです。アダプターに新しいリストを渡したくない場合は (渡さないように思われるため)、単に . の後に使用items.addAllしますclear()。これにより、アダプターが参照しているリストと同じリストを確実に変更できます。

于 2013-02-01T20:20:23.767 に答える
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()  
        }
});
于 2013-01-24T14:52:16.240 に答える
3
adpter.notifyDataSetInvalidated();

onPause()Activity クラスのメソッドでこれを試してください。

于 2013-02-01T13:01:14.827 に答える
1
adapter.setNotifyDataChanged()

トリックを行う必要があります。

于 2013-11-20T07:16:43.150 に答える
0

このようにしてみてください:

this.notifyDataSetChanged();

それ以外の:

adapter.notifyDataSetChanged();

アダプタークラスにnotifyDataSetChanged()しない必要があります。ListView

于 2013-01-31T11:02:28.480 に答える