这次主要是把 RecyclerView 比较常用的基本的点,在这里集中整理一下。从这篇文章主要梳理以下几点:

  • 优雅的实现:item 点击事件 & item 长点击事件
  • RecyclerView 添加 divider 的标准姿势
  • RecyclerView 实现 item 的拖曳排序和滑动删除
  • 拖曳排序时,限制首个 item 固定的实现

先看一下最终的效果图:

swipe and drag

drag

自从 RecyclerView 发布以来,由于其高度的可交互性被广泛使用。相信大家肯定对它的使用方法已经非常熟练了,今天主要是为大家总结一下较正常用法更加优雅的方式。

如果你想再回顾一下 RecyclerView 的基本使用方法,推荐鸿洋的这篇文章:
Android RecyclerView 使用完全解析 体验艺术般的控件

优雅的实现:item 点击事件 & item 长点击事件

使用方式

RecyclerView 的 api 虽然没有提供 onItemClickListener 但是提供了 addOnItemTouchListener() 方法,既然可以添加触摸监听,那么我们完全可以获取触摸手势来识别点击事件,然后通过触摸坐标来判断点击的是哪一个item。

mRecyclerView.addOnItemTouchListener(new OnRecyclerItemClickListener(mRecyclerView) {@Overridepublic void onItemClick(RecyclerView.ViewHolder viewHolder) {//TODO item 点击事件}@Overridepublic void onLongClick(RecyclerView.ViewHolder viewHolder) {//TODO item 长按事件}});

其中 OnRecyclerItemClickListener 是自定义的一个触摸监听器,代码如下:

public abstract class OnRecyclerItemClickListener implements RecyclerView.OnItemTouchListener{private GestureDetectorCompat mGestureDetectorCompat;//手势探测器private RecyclerView mRecyclerView;public OnRecyclerItemClickListener(RecyclerView recyclerView) {mRecyclerView = recyclerView;mGestureDetectorCompat = new GestureDetectorCompat(mRecyclerView.getContext(),new ItemTouchHelperGestureListener());}@Overridepublic boolean onInterceptTouchEvent(RecyclerView rv, MotionEvent e) {mGestureDetectorCompat.onTouchEvent(e);return false;}@Overridepublic void onTouchEvent(RecyclerView rv, MotionEvent e) {mGestureDetectorCompat.onTouchEvent(e);}@Overridepublic void onRequestDisallowInterceptTouchEvent(boolean disallowIntercept) {}public abstract void onItemClick(RecyclerView.ViewHolder viewHolder);public abstract void onLongClick(RecyclerView.ViewHolder viewHolder);
}

GestureDetectorCompat 中传入了一个 ItemTouchHelperGestureListener,代码如下:

private class ItemTouchHelperGestureListener extends GestureDetector.SimpleOnGestureListener{//一次单独的轻触抬起手指操作,就是普通的点击事件@Overridepublic boolean onSingleTapUp(MotionEvent e) {View childViewUnder = mRecyclerView.findChildViewUnder(e.getX(), e.getY());if (childViewUnder != null) {RecyclerView.ViewHolder childViewHolder = mRecyclerView.getChildViewHolder(childViewUnder);onItemClick(childViewHolder);}return true;}//长按屏幕超过一定时长,就会触发,就是长按事件@Overridepublic void onLongPress(MotionEvent e) {View childViewUnder = mRecyclerView.findChildViewUnder(e.getX(), e.getY());if (childViewUnder != null) {RecyclerView.ViewHolder childViewHolder = mRecyclerView.getChildViewHolder(childViewUnder);onLongClick(childViewHolder);}}}

原理分析

上面的代码很简单没什么复杂的地方,就是通过一个手势探测器 GestureDetectorCompat 来探测屏幕事件,然后通过手势监听器 SimpleOnGestureListener 来识别手势事件的种类,然后调用我们设置的对应的回调方法。这里值得说的是:当获取到了 RecyclerView 的点击事件和触摸事件数据 MotionEvent,那么如何才能知道点击的是哪一个 item 呢?

RecyclerView已经为我们提供了这样的方法:findChildViewUnder()

我们可以通过这个方法获得点击的 item ,同时我们调用 RecyclerView 的另一个方法 getChildViewHolder(),可以获得该 item 的 ViewHolder,最后再回调我们定义的虚方法 onItemClick() 就ok了,这样我们就可以在外部实现该方法来获得 item 的点击事件了。

RecyclerView 添加 divider 的标准姿势

当你想给条目间添加 divider 时,你可能自然而然的去尝试这种方式:

<android.support.v7.widget.RecyclerViewandroid:divider="#ffff0000"android:dividerHeight="10dp"android:layout_width="match_parent"android:layout_height="match_parent" />

其实 RecyclerView 是没有这两个属性的,就算你写上也不会有任何效果。
当然你还可以通过给 item 的最外层布局设置一个 margin 值,甚至你还可以专门在 item 布局中的适当地方添加一个高度/宽度为 1 的带背景的 View 作为 divider,这两种方法呢,确实有效果,但是不够优雅,有时还可能带来一些想不到的问题。

其实官方还是为我们提供了为 RecyclerView 添加分割线的方式的,那就是方法: mRecyclerView.addItemDecoration() 。该方法的参数为 RecyclerView.ItemDecoration,该类为抽象类,且官方目前并没有提供默认的实现类,我们只能自己来实现。

使用方式

列表布局的分割线实例:

public class DividerListItemDecoration extends RecyclerView.ItemDecoration {private static final int[] ATTRS = new int[]{android.R.attr.listDivider};public static final int HORIZONTAL_LIST = LinearLayoutManager.HORIZONTAL;public static final int VERTICAL_LIST = LinearLayoutManager.VERTICAL;private Drawable mDivider;private int mOrientation;public DividerListItemDecoration(Context context, int orientation) {final TypedArray a = context.obtainStyledAttributes(ATTRS);mDivider = a.getDrawable(0);a.recycle();setOrientation(orientation);}public DividerListItemDecoration(Context context, int orientation, int drawableId) {mDivider = ContextCompat.getDrawable(context, drawableId);setOrientation(orientation);}public void setOrientation(int orientation) {if (orientation != HORIZONTAL_LIST && orientation != VERTICAL_LIST) {throw new IllegalArgumentException("invalid orientation");}mOrientation = orientation;}//画线 > 就是画出你想要的分割线样式@Overridepublic void onDraw(Canvas c, RecyclerView parent) {if (mOrientation == VERTICAL_LIST) {drawVertical(c, parent);} else {drawHorizontal(c, parent);}}public void drawVertical(Canvas c, RecyclerView parent) {final int left = parent.getPaddingLeft();final int right = parent.getWidth() - parent.getPaddingRight();final int childCount = parent.getChildCount();for (int i = 0; i < childCount; i++) {final View child = parent.getChildAt(i);android.support.v7.widget.RecyclerView v = new android.support.v7.widget.RecyclerView(parent.getContext());final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams();final int top = child.getBottom() + params.bottomMargin;final int bottom = top + mDivider.getIntrinsicHeight();mDivider.setBounds(left, top, right, bottom);mDivider.draw(c);}}public void drawHorizontal(Canvas c, RecyclerView parent) {final int top = parent.getPaddingTop();final int bottom = parent.getHeight() - parent.getPaddingBottom();final int childCount = parent.getChildCount();for (int i = 0; i < childCount; i++) {final View child = parent.getChildAt(i);final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams();final int left = child.getRight() + params.rightMargin;final int right = left + mDivider.getIntrinsicHeight();mDivider.setBounds(left, top, right, bottom);mDivider.draw(c);}}//设置条目周边的偏移量@Overridepublic void getItemOffsets(Rect outRect, int itemPosition, RecyclerView parent) {if (mOrientation == VERTICAL_LIST) {outRect.set(0, 0, 0, mDivider.getIntrinsicHeight());} else {outRect.set(0, 0, mDivider.getIntrinsicWidth(), 0);}}
}

网格布局分割线实例:

public class DividerGridItemDecoration extends RecyclerView.ItemDecoration {private static final int[] ATTRS = new int[]{android.R.attr.listDivider};private Drawable mDivider;private int lineWidth = 1;public DividerGridItemDecoration(Context context) {final TypedArray a = context.obtainStyledAttributes(ATTRS);mDivider = a.getDrawable(0);a.recycle();}public DividerGridItemDecoration(int color) {mDivider = new ColorDrawable(color);}public DividerGridItemDecoration() {this(Color.parseColor("#cccccc"));}@Overridepublic void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) {drawHorizontal(c, parent);drawVertical(c, parent);}private int getSpanCount(RecyclerView parent) {// 列数int spanCount = -1;RecyclerView.LayoutManager layoutManager = parent.getLayoutManager();if (layoutManager instanceof GridLayoutManager) {spanCount = ((GridLayoutManager) layoutManager).getSpanCount();} else if (layoutManager instanceof StaggeredGridLayoutManager) {spanCount = ((StaggeredGridLayoutManager) layoutManager).getSpanCount();}return spanCount;}public void drawHorizontal(Canvas c, RecyclerView parent) {int childCount = parent.getChildCount();for (int i = 0; i < childCount; i++) {final View child = parent.getChildAt(i);final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams();final int left = child.getLeft() - params.leftMargin;final int right = child.getRight() + params.rightMargin+ lineWidth;final int top = child.getBottom() + params.bottomMargin;final int bottom = top + lineWidth;mDivider.setBounds(left, top, right, bottom);mDivider.draw(c);}}public void drawVertical(Canvas c, RecyclerView parent) {final int childCount = parent.getChildCount();for (int i = 0; i < childCount; i++) {final View child = parent.getChildAt(i);final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams();final int top = child.getTop() - params.topMargin;final int bottom = child.getBottom() + params.bottomMargin;final int left = child.getRight() + params.rightMargin;final int right = left + lineWidth;mDivider.setBounds(left, top, right, bottom);mDivider.draw(c);}}private boolean isLastColum(RecyclerView parent, int pos, int spanCount, int childCount) {RecyclerView.LayoutManager layoutManager = parent.getLayoutManager();if (layoutManager instanceof GridLayoutManager) {if ((pos + 1) % spanCount == 0)// 如果是最后一列,则不需要绘制右边{return true;}} else if (layoutManager instanceof StaggeredGridLayoutManager) {int orientation = ((StaggeredGridLayoutManager) layoutManager).getOrientation();if (orientation == StaggeredGridLayoutManager.VERTICAL) {if ((pos + 1) % spanCount == 0)// 如果是最后一列,则不需要绘制右边{return true;}} else {childCount = childCount - childCount % spanCount;if (pos >= childCount)// 如果是最后一列,则不需要绘制右边return true;}}return false;}private boolean isLastRaw(RecyclerView parent, int pos, int spanCount, int childCount) {LayoutManager layoutManager = parent.getLayoutManager();if (layoutManager instanceof GridLayoutManager) {childCount = childCount - childCount % spanCount;if (pos >= childCount)// 如果是最后一行,则不需要绘制底部return true;} else if (layoutManager instanceof StaggeredGridLayoutManager) {int orientation = ((StaggeredGridLayoutManager) layoutManager).getOrientation();// StaggeredGridLayoutManager 且纵向滚动if (orientation == StaggeredGridLayoutManager.VERTICAL) {childCount = childCount - childCount % spanCount;// 如果是最后一行,则不需要绘制底部if (pos >= childCount)return true;} else// StaggeredGridLayoutManager 且横向滚动{// 如果是最后一行,则不需要绘制底部if ((pos + 1) % spanCount == 0) {return true;}}}return false;}@Overridepublic void getItemOffsets(Rect outRect, View view, RecyclerView parent, State state) {boolean b = state.willRunPredictiveAnimations();int itemPosition = ((RecyclerView.LayoutParams) view.getLayoutParams()).getViewLayoutPosition();int spanCount = getSpanCount(parent);int childCount = parent.getAdapter().getItemCount();
//        if (isLastRaw(parent, itemPosition, spanCount, childCount))// 如果是最后一行,则不需要绘制底部
//        {
//            outRect.set(0, 0, lineWidth, 0);
//        }
//        else if (isLastColum(parent, itemPosition, spanCount, childCount))// 如果是最后一列,则不需要绘制右边
//        {if (b){outRect.set(0, 0, lineWidth, lineWidth);}else {
//                outRect.set(0, 0, 0, lineWidth);}
//        }
//        else {outRect.set(0, 0, lineWidth, lineWidth);
//        }}
}

使用说明

上面给出的两个实例都是最简单的一条线的分割。这里的分割线你是可以自由的去自定义它的,具体如何实现也不是太复杂,这里不再做详细介绍了,推荐一篇文章:

RecyclerView之ItemDecoration 讲解及高级特性实践

RecyclerView 实现 item 的拖曳排序和滑动删除

下面就主要为大家梳理一下拖曳排序和滑动删除的实现,具体实现效果看文章首部效果图,这里就不再重复放图了。

实现方式

主要就要使用到 ItemTouchHelper,ItemTouchHelper 一个帮助开发人员处理拖拽和滑动删除的实现类,它能够让你非常容易实现侧滑删除、拖拽的功能。(ItemTouchHelper 的使用并不仅仅局限于 RecyclerView 的滑动删除,你同意可以用在其他需要拖曳滑动的地方。当然,今天我们不涉及其他地方的使用)

实现的代码并关联到 RecyclerView 非常简单,代码如下:

ItemTouchHelper itemTouchHelper = new ItemTouchHelper(new ItemTouchHelper.Callback());
itemTouchHelper.attachToRecyclerView(mRecyclerView);

代码很简单,没什么好说的。需要我们关注的是创建 ItemTouchHelper 时传入的参数 ItemTouchHelper.Callback() 。ItemTouchHelper 会在拖拽的时候回调 Callback 中相应的方法,我们只需在 Callback 中实现自己的逻辑。

自定义一个类继承实现 ItemTouchHelper.Callback 接口,需要实现以下方法:

//通过返回值来设置是否处理某次拖曳或者滑动事件
public abstract int getMovementFlags(RecyclerView recyclerView,ViewHolder viewHolder);//当长按并进入拖曳状态时,拖曳的过程中不断的回调此方法
public abstract boolean onMove(RecyclerView recyclerView,ViewHolder viewHolder, ViewHolder target);//滑动删除的回调
public abstract void onSwiped(ViewHolder viewHolder, int direction);

getMovementFlags() 用于设置是否处理拖拽事件和滑动事件,以及拖拽和滑动操作的方向,有以下两种情况:

  • 如果是列表类型的 RecyclerView,拖拽只有 UP、DOWN 两个方向
  • 如果是网格类型的则有 UP、DOWN、LEFT、RIGHT 四个方向

该方法需要编写的代码如下:

@Override
public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {if (recyclerView.getLayoutManager() instanceof GridLayoutManager) {int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN |ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;int swipeFlags = 0;return makeMovementFlags(dragFlags, swipeFlags);} else {int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN;int swipeFlags = 0;return makeMovementFlags(dragFlags, swipeFlags);}
}

dragFlags 是拖拽标志,
swipeFlags 是滑动标志,
swipeFlags 都设置为0,暂时不考虑滑动相关操作。

如果设置了相关的 dragFlags,那么当长按 item 的时候就会进入拖拽并在拖拽过程中不断回调 onMove() 方法,我们就在这个方法里获取当前拖拽的 item 和已经被拖拽到所处位置的 item 的ViewHolder,有了这2个 ViewHolder,我们就可以交换他们的数据集并调用 Adapter 的notifyItemMoved 方法来刷新 item。

@Override
public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder, RecyclerView.ViewHolder target) {//拖动的 item 的下标int fromPosition = viewHolder.getAdapterPosition();//目标 item 的下标,目标 item 就是当拖曳过程中,不断和拖动的 item 做位置交换的条目。int toPosition = target.getAdapterPosition();if (fromPosition < toPosition) {for (int i = fromPosition; i < toPosition; i++) {Collections.swap(((RecyAdapter) mAdapter).getDataList(), i, i + 1);}} else {for (int i = fromPosition; i > toPosition; i--) {Collections.swap(((RecyAdapter) mAdapter).getDataList(), i, i - 1);}}mAdapter.notifyItemMoved(fromPosition, toPosition);return true;
}

只要重写完上面这两个方法,RecyclerView 就能实现拖曳的效果了。是不是很简单?但是虽然拖曳是没什么问题了,但是并不能达到下图的效果,因为你正在拖曳的 item 并没有阴影效果。

拖曳 item

那怎么才能实现被拖曳的 item 有背景颜色加深起到强调的视觉效果呢?这是需要重写下面两个方法:

//当长按 item 刚开始拖曳的时候调用
@Override
public void onSelectedChanged(RecyclerView.ViewHolder viewHolder, int actionState) {if (actionState != ItemTouchHelper.ACTION_STATE_IDLE) {//给被拖曳的 item 设置一个深颜色背景viewHolder.itemView.setBackgroundColor(Color.LTGRAY);}super.onSelectedChanged(viewHolder, actionState);
}//当完成拖曳手指松开的时候调用
@Override
public void clearView(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {super.clearView(recyclerView, viewHolder);//给已经完成拖曳的 item 恢复开始的背景。//这里我们设置的颜色尽量和你 item 在 xml 中设置的颜色保持一致viewHolder.itemView.setBackgroundColor(Color.WHITE);
}

这样就能完全达到上面图片的效果了。

滑动删除

如何实现滑动删除呢?我们只需要实现第三个方法 onSwipe() 就行了。代码如下:

@Override
public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {int adapterPosition = viewHolder.getAdapterPosition();mAdapter.notifyItemRemoved(adapterPosition);((RecyAdapter) mAdapter).getDataList().remove(adapterPosition);
}

同时也不要忘了修改一下 getMovementFlags() 方法,以便能够相应滑动事件:

@Override
public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {if (recyclerView.getLayoutManager() instanceof GridLayoutManager) {int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN |ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;int swipeFlags = 0;return makeMovementFlags(dragFlags, swipeFlags);} else {int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN;//注意:和拖曳的区别就是在这里int swipeFlags = ItemTouchHelper.START | ItemTouchHelper.END;return makeMovementFlags(dragFlags, swipeFlags);}
}

那目前你就能完美的实现拖曳排序和滑动删除了。

拖曳排序,首个固定

有时我们希望首个 item 不能被拖曳排序。比如我们在新闻 App 中常见当我们进行新闻分类时,“热门”新闻这个分类总是第一个且不能被拖曳修改,类似下面的效果:

首个固定不能被拖曳

那么怎么才能达到上面的效果呢?在上面我们的 Callback 类中有一个方法:

public boolean isLongPressDragEnabled() {return true;
}

这个方法是为了告诉 ItemTouchHelper 是否需要 RecyclerView 支持长按拖拽,默认返回是 ture,理所当然我们要支持,所以我们没有重写,因为默认true。但是这样做是默认全部的item都可以拖拽,怎么实现部分item拖拽呢,在 isLongPressDragEnabled 方法的源码中有提示说,如果想自定义拖曳 view,那么就使用 startDrag(ViewHolder) 方法。

第一步:那么我们就先重写 isLongPressDragEnabled() 方法,返回 false 让它控制所有的 item 都不能拖曳。

public boolean isLongPressDragEnabled() {return false;
}

第二步:我们给 RecyclerView 设置 item 的长按监听事件,然后判断这个 item 是不是第一个(或者最后一个,如果你不想让最后一个被拖曳的话),如果不是我们就手动调用 startDrag(ViewHolder) 让 item 开始被拖曳。
结合上面我们提供的给 item 设置点击和长按事件的方法,我们可以这样:

mRecyclerView.addOnItemTouchListener(new OnRecyclerItemClickListener(mRecyclerView) {@Overridepublic void onItemClick(RecyclerView.ViewHolder viewHolder) {//TODO:点击事件}@Overridepublic void onLongClick(RecyclerView.ViewHolder viewHolder) {//当 item 被长按且不是第一个时,开始拖曳这个 itemif (viewHolder.getLayoutPosition() != 0) {itemTouchHelper.startDrag(viewHolder);}}
});

第三步:如果你以为上面两步你就达到首个 item 固定不被拖曳的话,恭喜你,答对了!首个 item 确实固定不能被拖曳了,可是看看下图,就会令你大跌眼睛:

首个固定,不能被拖曳,却能被挤动

虽然我们通过上面两步控制了首个 item 不能被长按拖曳,但是我们并没有处理,别的 item 被拖曳到首个 item 的情况。那么如何才能让首个 item 不被挤掉呢,这个也很简单,只需要在 Callback 的 onMove() 方法中处理首个 item 被当着目标 item 的情况就行了。

@Override
public boolean onMove(...) {int fromPosition = viewHolder.getAdapterPosition();int toPosition = target.getAdapterPosition();//其他地方代码都和上面的一样,这个就直接省略了//这里判断如果目标 item 是首个 item,那么就直接返回false,表示不响应此次拖曳移动if (toPosition == 0) {return false;}...return true;
}

好了,到这里就大功告成了。

本文源代码地址:https://github.com/OCNYang/RecyclerViewEvent

参考文章:
http://chuansong.me/n/400690551872
http://chuansong.me/n/400690851058
http://www.10tiao.com/html/227/201705/2650239745/1.html

RecyclerView 梳理:点击amp;长按事件、分割线、拖曳排序、滑动删除相关推荐

  1. html5实现ios长按图标后进入图标排序及删除功能的效果

    2019独角兽企业重金招聘Python工程师标准>>> html5实现ios长按图标后进入图标排序及删除功能的效果 我们知道在ios(国产定制安卓系统基本都有)设备上按下图标,图标就 ...

  2. Android 让View 和 ViewGroup 同时响应点击或者长按事件

    面试的时候,被问到如何让View 和 其 父View 同时响应长按事件. 我还记得当时自己的回答,子View 里面 处理了长按事件,但是返回了false, 没有处理,然后会继续调用到父View 的长按 ...

  3. android.support.v7 fragme,打造最强RecyclerView侧滑菜单,长按拖拽Item,滑动删除Item

    前几天写了一片关于RecyclerView滑动删除Item,RecyclerView长按拖拽Item的博客,本来很简单一个使用,阅读量还挺高的,原博客传送门. 今天介绍一个RecyclerView I ...

  4. Android手机红外开发—点击和长按事件

    上一篇文章<红外技术及Android手机红外遥控器开发>中简单介绍了红外遥控技术和Android 手机红外遥控开发操控中的点击事件,只需要按照编码协议规则对键值等进行编码解析,最后转化成数 ...

  5. RecyclerView 梳理:点击长按事件、分割线、拖曳排序、滑动删除

    本文作者 作者:OCNYang 链接:http://www.jianshu.com/p/70788a7a5547 本文由作者投稿发布. 这次主要是把 RecyclerView 比较常用的基本的点,在这 ...

  6. [Unity UGUI]点击和长按组件

    需求 游戏项目中卡片经常需要按钮/卡片的点击或者长按事件,这里提供一个好用的组件. 组件 using UnityEngine; using UnityEngine.Events; using Unit ...

  7. 微信小程序长按事件触发点击事件的BUG处理

    微信小程序开发说实话还是有点糟心的,经过事件冒泡的坑之后,又遇到了长按事件(longtap)必触发点击事件(tap)的BUG 实例代码 wxml <view class="contai ...

  8. Compose 手势事件:防止重复点击,双击,长按,全局触摸隐藏键盘

    前言 JetPack Compose (后续简称compose) release版已经出来了三四个月了,虽然没正式版之前也学过几次,但一直没有机会用,在加上api的变更,导致之前学的都忘完了,现在终于 ...

  9. android listview item长按事件,ListView的点击事件,长按事件,数据刷新

    1.刷新数据 当List中的数据发生改变时,使用listViewAdapter.notifyDataSetChanged();通知Adapter刷新数据. 在Activity中找到ListView,为 ...

最新文章

  1. pdo mysql防注入_Php中用PDO查询Mysql来避免SQL注入风险的方法
  2. debian php redis,debian7.7环境下 源码安装redis以及phpredis扩展
  3. 使用Docker,Chef和Amazon OpsWorks进行集群范围的Java / Scala应用程序部署
  4. python中numpy是什么_什么是NumPy?
  5. 刘庆敏 博客linux,Linux内核源码分析--zImage出生实录(Linux-3.0 ARMv7)
  6. alook浏览器js扩展网站_备用浏览器Alook所能实现的功能果真强大(文末有alook兑换码抽奖)...
  7. 软件安装下载的镜像站、国内源
  8. c++学习---继承与派生类
  9. .NET程序员应掌握的常用类库
  10. Java如何连接数据库
  11. Java权限管理系统源代码下载
  12. mysql常用增删改查sql语句
  13. 量子计算机窦文涛,中国式浪漫,美到哭泣!
  14. 小程序渲染层网络层错误404
  15. 工具使用 - office快捷键
  16. op积分上反馈电阻 作用 理解 op 积分
  17. (大厂必备)厂长熬夜爆肝万字之多线程高并发JUC编程(一)⭐学妹已收藏
  18. P41-前端基础-clear解决受浮动元素的影响而改变位置
  19. 1800勘误表_专业的ASP.NET 2.0勘误表
  20. shell学习笔记---工具学习

热门文章

  1. Springboot+采用协同过滤算法的视频推荐系统的设计与实现 毕业设计-附源码261620
  2. trackmaker翻译_水工词汇翻译
  3. 求职简历撰写要点和模板分享
  4. Windows输入法注入原理
  5. cyclone v QSPI烧写问题(not able to map flash id)
  6. 日本雅虎乐天商城批量上传步骤
  7. 【ZOJ题目分类】备忘
  8. 埃森哲java开发怎么样_花费2亿,耗时2年,网站还没建完,咨询公司埃森哲被告上法庭!...
  9. 适配器模式_21世纪的设计模式:适配器模式
  10. [转]Go在谷歌:以软件工程为目的的语言设计