这是效果图:
activity_main.xml
01 |
< LinearLayout xmlns:android = "http://schemas.android.com/apk/res/android" |
02 |
xmlns:tools = "http://schemas.android.com/tools" |
03 |
android:layout_width = "match_parent" |
04 |
android:layout_height = "match_parent" |
08 |
android:layout_width = "wrap_content" |
09 |
android:layout_height = "wrap_content" |
10 |
android:id = "@+id/listview" |
listview_item.xml listview每一条
01 |
< LinearLayout xmlns:android = "http://schemas.android.com/apk/res/android" |
02 |
xmlns:tools = "http://schemas.android.com/tools" |
03 |
android:layout_width = "match_parent" |
04 |
android:layout_height = "match_parent" |
05 |
android:paddingBottom = "@dimen/activity_vertical_margin" |
06 |
android:paddingLeft = "@dimen/activity_horizontal_margin" |
07 |
android:paddingRight = "@dimen/activity_horizontal_margin" |
08 |
android:paddingTop = "@dimen/activity_vertical_margin" > |
11 |
android:layout_width = "wrap_content" |
12 |
android:layout_height = "wrap_content" |
13 |
android:id = "@+id/text" /> |
footer.xml 作为每一页翻到最后时显示:正在加载中.....
01 |
< LinearLayout xmlns:android = "http://schemas.android.com/apk/res/android" |
02 |
xmlns:tools = "http://schemas.android.com/tools" |
03 |
android:layout_width = "match_parent" |
04 |
android:layout_height = "match_parent" |
05 |
android:orientation = "horizontal" |
06 |
android:paddingBottom = "@dimen/activity_vertical_margin" |
07 |
android:paddingLeft = "@dimen/activity_horizontal_margin" |
08 |
android:paddingRight = "@dimen/activity_horizontal_margin" |
09 |
android:paddingTop = "@dimen/activity_vertical_margin" > |
12 |
style = "?android:attr/process" |
13 |
android:layout_width = "50dp" |
14 |
android:layout_height = "wrap_content" /> |
17 |
android:layout_width = "match_parent" |
18 |
android:layout_height = "match_parent" |
19 |
android:text = "正在加载....." |
20 |
android:textSize = "18dp" /> |
下面是主界面:
01 |
public class MainActivity extends Activity { |
03 |
private ListView listview; |
04 |
private List<String> data = new ArrayList<String>(); |
05 |
private ArrayAdapter<String> adapter; |
06 |
private LayoutInflater inflater; |
07 |
private View footer; // 页脚-正在加载中..... |
10 |
protected void onCreate(Bundle savedInstanceState) { |
11 |
super .onCreate(savedInstanceState); |
12 |
setContentView(R.layout.activity_main); |
13 |
inflater = getLayoutInflater(); |
14 |
footer = inflater.inflate(R.layout.footer, null ); |
15 |
listview = (ListView) findViewById(R.id.listview); |
16 |
listview.setOnScrollListener( new scrollListener()); |
17 |
data.addAll(DataService.getData( 0 , 20 )); |
18 |
adapter = new ArrayAdapter<String>( this , R.layout.listview_item, |
20 |
/* 在适配器之前加页脚,这样适配器会重新被封装成 '有页脚的适配器' */ |
21 |
listview.addFooterView(footer); |
22 |
listview.setAdapter(adapter); |
23 |
listview.removeFooterView(footer); |
30 |
public class scrollListener implements OnScrollListener { |
32 |
int pagesize = 20;// 每页显示条目 |
33 |
int maxpage = 5;// 最多页数 |
34 |
int currentpage;// 当前页 |
36 |
boolean finish_load = true;// 加载是否完成,默认完成 |
39 |
* 监听滚动状态改变:1-手指正在滑动 2-手指停止滑动 3-组件停止滚动 |
41 |
public void onScrollStateChanged(AbsListView view, int scrollState) { |
45 |
* firstVisibleItem:第一个可见item visibleItemCount:可见item数量 |
46 |
* totalItemCount:总条目数量 |
48 |
public void onScroll(AbsListView view, int firstVisibleItem, |
49 |
int visibleItemCount, int totalItemCount) { |
50 |
final int total = totalItemCount; |
52 |
if (listview.getLastVisiblePosition() + 1 == totalItemCount) { |
53 |
if (totalItemCount > 0) { |
55 |
currentpage = totalItemCount % pagesize == 0 ? totalItemCount |
57 |
: totalItemCount / pagesize + 1; |
58 |
nextpage = currentpage + 1; |
60 |
* 如果当前页小于规定的最大页数,并且加载完成(不断滚动就会不断执行onScroll方法, |
63 |
if (nextpage <= maxpage && finish_load) { |
66 |
listview.addFooterView(footer); |
68 |
new Thread(new Runnable() { |
72 |
} catch (InterruptedException e) { |
75 |
List<String> l = DataService.getData(total, |
77 |
handle.sendMessage(handle.obtainMessage(123, l)); |
86 |
/* 通过handle和主线程通讯,主线程接收消息更新UI */ |
87 |
Handler handle = new Handler() { |
88 |
public void handleMessage(Message msg) { |
89 |
data.addAll((List<String>) msg.obj); |
90 |
adapter.notifyDataSetChanged(); |
92 |
if (listview.getFooterViewsCount() > 0 ) |
93 |
listview.removeFooterView(footer); |
获取数据service (模拟从网上获取)
01 |
public class DataService { |
04 |
* @param startposition |
05 |
* :从第startposition条数据开始加载 |
10 |
public static List<String> getData( int startposition, int pagesize) { |
11 |
List<String> list = new ArrayList<String>(); |
12 |
for ( int i = startposition; i <startposition+ pagesize; i++) { |
13 |
list.add( "第" + i + "条数据" ); |
下面是Handle详解:
一、Handler的定义:
主要接受子线程发送的数据, 并用此数据配合主线程更新UI.
解释: 当应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的UI控件,
进行事件分发, 比如说, 你要是点击一个 Button ,Android会分发事件到Button上,来响应你的操作。
如果此时需要一个耗时的操作,例如: 联网读取数据,
或者读取本地较大的一个文件的时候,你不能把这些操作放在主线程中,,如果你放在主线程中的话,界面会出现假死现象,
如果5秒钟还没有完成的话,,会收到Android系统的一个错误提示 "强制关闭".
这个时候我们需要把这些耗时的操作,放在一个子线程中,因为子线程涉及到UI更新,,Android主线程是线程不安全的,
也就是说,更新UI只能在主线程中更新,子线程中操作是危险的. 这个时候,Handler就出现了.,来解决这个复杂的问题 ,
由于Handler运行在主线程中(UI线程中), 它与子线程可以通过Message对象来传递数据,
这个时候,Handler就承担着接受子线程传过来的(子线程用sedMessage()方法传弟)Message对象,(里面包含数据) ,
把这些消息放入主线程队列中,配合主线程进行更新UI。
二、Handler一些特点
handler可以分发Message对象和Runnable对象到主线程中, 每个Handler实例,都会绑定到创建他的线程中(一般是位于主线程),
它有两个作用: (1): 安排消息或Runnable 在某个主线程中某个地方执行, (2)安排一个动作在不同的线程中执行
Handler中分发消息的一些方法
post(Runnable)
postAtTime(Runnable,long)
postDelayed(Runnable long)
sendEmptyMessage(int)
sendMessage(Message)
sendMessageAtTime(Message,long)
sendMessageDelayed(Message,long)
以上post类方法允许你排列一个Runnable对象到主线程队列中,
sendMessage类方法, 允许你安排一个带数据的Message对象到队列中,等待更新.
- Android ListView滑动底部自动加载更多
直接上代码: // lv = (ListView) findViewById(R.id.lv); // // for(int i = 0;i < 50;i++){ // ls.add(" ...
- android 在自定义的listview(有刷新加载项)列表中,数据过少时不能铺满整个屏幕时,header和footer同时显示问题
android 在自定义的listview(有刷新加载项)列表中,数据过少时,当刷新时,加载项也会显示,这是很头疼的一个问题,查阅了一些资料,总结了一个比较不错的方法: 原来代码: @Overrid ...
- Android中自定义ListView实现上拉加载更多和下拉刷新
ListView是Android中一个功能强大而且很常用的控件,在很多App中都有ListView的下拉刷新数据和上拉加载更多这个功能.这里我就简单记录一下实现过程. 实现这个功能的方法不止一个,Gi ...
- Android动画之仿美团加载数据等待时,小人奔跑进度动画对话框(附顺丰快递员奔跑效果)
Android动画之仿美团加载数据等待时,小人奔跑进度动画对话框(附顺丰快递员奔跑效果) 首句依然是那句老话,你懂得! finddreams :(http://blog.csdn.net/finddr ...
- 利用WPF的ListView进行大数据量异步加载
原文:利用WPF的ListView进行大数据量异步加载 由于之前利用Winform的ListView进行大数据量加载的时候,诟病良多,所以今天试着用WPF的ListView来做了一下,结果没有让我失望 ...
- php+ajax实现拖动滚动条分批加载请求加载数据
HTML: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w ...
- Android引入高速缓存的异步加载全分辨率
Android引进高速缓存的异步加载全分辨率 为什么要缓存 通过图像缩放,我们这样做是对的异步加载优化的大图,但现在的App这不仅是一款高清大图.图.动不动就是图文混排.以图代文,假设这些图片都载入到 ...
- json解析,异步下载(listview仅滑动时加载)Demo总结
异步加载的练习demo 主要涉及知识点: 1.解析json格式数据,主要包括图片,文本 2.使用AsynTask异步方式从网络下载图片 3.BaseAdapter的"优雅"使用 4 ...
- Android ViewPager Fragment使用懒加载提升性能
Android ViewPager Fragment使用懒加载提升性能 Fragment在如今的Android开发中越来越普遍,但是当ViewPager结合Fragment时候,由于Androi ...
随机推荐
- Spring MVC 4.2 CORS 跨域访问
跨站 HTTP 请求(Cross-site HTTP request)是指发起请求的资源所在域不同于该请求所指向资源所在的域的 HTTP 请求.比如说,域名A(http://domaina.examp ...
- AngularJS driective 封装 自动滚动插件
1.ui-smooth-scroll.js文件内容 angular.module('app') .directive('uiSmoothScroll', ['$location', '$anchorS ...
- 【转】php里面也可以使用协程
原文链接:http://blog.51cto.com/chinalx1/2089327 http://nikic.github.io/2012/12/22/Cooperative-multitaski ...
- eclipse全屏模式
- 内容匹配广告投放技术4:网盟CTR预估(百度文库课程)
原文:http://wbj0110.iteye.com/blog/2043065 该文是百度文库课程<计算广告学之内容匹配广告&展示广告原理.技术和实践>的课程笔记,感谢百度! 课 ...
- C++中用完需要释放掉内存的几个类
BSTR BSTR bstrXML = NULL; //用完以后,或者 catch段中 if(bstrXML) ::SysFreeString(result); VARIANT VARIANT v ...
- 【Python】在控制台输出不同颜色的文字
今天调程序出了一个极为奇怪的问题,由于控制台输出挺多,就想把问题着重表示一下,具体即是在控制台输出红色文字. 于是在网上搜寻到了这篇:https://www.cnblogs.com/gongxr/p/ ...
- Jquery 应用积累
1.控制div显隐 $("#id").show()表示display:block, $("#id").hide()表示display:none; $(" ...
- Gradle修改Maven仓库地址
博客已经搬家https://www.tianmingxing.com 背景 不知从什么时候大家开始使用gradle管理项目了,随着时间的推移从maven转过来的人肯定越来越多.关于gradle的优势在 ...
- Oracle的REGEXP_SUBSTR函数简单使用方法
REGEXP_SUBSTR延伸SUBSTR函数的功能.让你搜索一个正則表達式模式字符串. 这也相似于REGEXP_INSTR.而是返回子字符串的位置,它返回的子字符串本身. 语法 Oracle数据库中 ...