异步删除ListView / RecyclerView项的最佳做法

时间:2016-04-03 19:30:00

标签: android database listview asynchronous android-recyclerview

我有ListView CursorLoader。用户可以打开ListView项(打开另一个Fragment)或删除项目。所有数据库操作都是异步发生的,通常只需要几分之一秒。但从技术上讲,用户可以删除一个项目,然后在删除回调和原因和错误之前打开该项目。处理这个问题的最佳方法是什么?以下是我看到的选项。

  1. 假设AsyncTask总是能够快速发生以避免出现问题
  2. 在UI线程上执行数据库操作
  3. ListView之前使AsyncTask无效(但这会导致用户界面中的闪存)
  4. AsyncTask
  5. 期间以某种方式阻止用户输入

    修改:我最终使用RecyclerView,但在我从数据库中删除该项目之前,我无法调用adapter.notifyItemRemoved(itemPos)

3 个答案:

答案 0 :(得分:6)

您的情况下的最佳选择是使用RecyclerView。因为当您点击删除时,您可以拨打adapter.notifyItemRemoved(itemPos)。因此,列表项将被移除RecyclerView中的动画,您不必担心删除操作结果。

请参阅Creating Lists and Cards上的Android Developer教程:

  

如果您有数据集,请使用RecyclerView小部件   元素在运行时根据用户操作或网络事件而变化。

以下是一个例子:

public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> implements View.OnClickListener, View.OnLongClickListener {

private ArrayList<String> mDataset;
private static Context sContext;

public MyAdapter(Context context, ArrayList<String> myDataset) {
    mDataset = myDataset;
    sContext = context;
}

@Override
public MyAdapter.ViewHolder onCreateViewHolder(ViewGroup parent,int viewType) {
    View v = LayoutInflater.from(parent.getContext()).inflate(R.layout.my_text_view, parent, false);

    ViewHolder holder = new ViewHolder(v);
    holder.mNameTextView.setOnClickListener(MyAdapter.this);

    holder.mNameTextView.setTag(holder);

    return holder;
}

@Override
public void onBindViewHolder(ViewHolder holder, int position) {

    holder.mNameTextView.setText(mDataset.get(position));

}

@Override
public int getItemCount() {
    return mDataset.size();
}


@Override
public void onClick(View view) {
    ViewHolder holder = (ViewHolder) view.getTag();
    if (view.getId() == holder.mNameTextView.getId()) {
        **//Important !!!**
        notifyItemRemoved(getPosition());
        // Make your database operation also here
    }
}



public static class ViewHolder extends RecyclerView.ViewHolder {
    public TextView mNumberRowTextView;
    public TextView mNameTextView;


    public ViewHolder(View v) {
        super(v);

        mNameTextView = (TextView) v.findViewById(R.id.nameTextView);
    }
}

答案 1 :(得分:3)

attribute: (UIDeviceOrientationIsPortrait(UIDevice.currentDevice().orientation) ? .Height : .Width)

或者使用带有移除动画的RecyclerView。

<强>的build.gradle

textField

<强> activity_main.xml中

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        final View itemView = convertView != null ? convertView : /* inflate view */;
        itemView.findViewById(R.id.delete_button).setOnClickListener(
                new View.OnClickListener() {
                    @Override
                    public void onClick(View view) {
                        itemView.setEnabled(false);
                        // OR view.setClickable(false);
                    }
                });
        return itemView;
    }

<强> my_item.xml

compile 'com.android.support:recyclerview-v7:23.2.1'

<强> MainActivity.java

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/my_recycler_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layoutManager="android.support.v7.widget.LinearLayoutManager" />
</RelativeLayout>

答案 2 :(得分:0)

如果您的数据库查询没有返回大量数据,则在UI线程上执行数据库操作没有问题。