一般来说,ListView的列表项都会采用相同的布局,只是填充的内容不同而已,这种情况下,Android提供了convertView帮我们缓存列表项,达到循环利用的目的,开发者也会使用ViewHolder模式来对ListView进行优化。但有的情况下,ListView的列表项布局是不尽相同的,甚至差别很大,这时就不能简单的使用同一个布局资源文件来表示不同类型的列表项了,而是必须区别对待。典型的情况如Android状态通知栏,如下图所示。

360安全卫士、唱吧,闪推这三款应用在状态通知栏的布局差别很大,那么我们又想使用同一个Adapter来表示不同列表项,这时就需要在Adapter中使用容器来包含这些不同的View了。这里定义SackOfViewAdapter类,继承BaseAdapter,在该类中定义容器变量mViewList来存储ListView中不同列表项的view;同时,定义两个构造函数,一个是在参数中指定容器的大小,并填充null值;一个则直接传入ListView列表项中所有view的集合。代码如下:

SackOfViewAdapter类的关键代码在newView函数和getView函数,其中newView函数用于给容器变量mViewList中值为null的元素赋值,一般由SackOfViewAdapter的子类实现。而getView函数重写自BaseAdapter类,是Adapter返回View类实例的关键函数,代码如下所示:

SackOfViewAdapter类的完整代码如下所示:

  1. public class SackOfViewsAdapter extends BaseAdapter {
  2. private List<View> mViewList = null;
  3. /**
  4. * 构造大小为count,值为null的view集合,这时需要子类重写newView函数
  5. */
  6. public SackOfViewsAdapter(int count) {
  7. super();
  8. mViewList = new ArrayList<View>(count);
  9. for (int i = 0; i < count; i++) {
  10. mViewList.add(null);
  11. }
  12. }
  13. /**
  14. * 由传入的view集合直接给容器赋值,如果view集合中有为null值的view,则子类必须实现newView函数
  15. */
  16. public SackOfViewsAdapter(List<View> views) {
  17. super();
  18. this.mViewList = views;
  19. }
  20. /**
  21. * 返回对应位置的列表项
  22. */
  23. @Override
  24. public Object getItem(int position) {
  25. return mViewList.get(position);
  26. }
  27. /**
  28. * 返回列表项的个数
  29. */
  30. @Override
  31. public int getCount() {
  32. return mViewList.size();
  33. }
  34. /**
  35. * getView函数创建的列表项类型个数
  36. */
  37. @Override
  38. public int getViewTypeCount() {
  39. return getCount();
  40. }
  41. /**
  42. * getView函数创建的view类型值,这里以view所在的位置作为类型值
  43. */
  44. @Override
  45. public int getItemViewType(int position) {
  46. return position;
  47. }
  48. /**
  49. * 如果Adapter中所有列表项都是可点击和可选择的,则返回true
  50. */
  51. @Override
  52. public boolean areAllItemsEnabled() {
  53. return false;
  54. }
  55. /**
  56. * 如果position所指的列表项不是分隔符,则返回true
  57. */
  58. @Override
  59. public boolean isEnabled(int position) {
  60. return false;
  61. }
  62. /**
  63. * 返回指定位置position的列表项的view
  64. */
  65. @Override
  66. public View getView(int position, View convertView, ViewGroup parent) {
  67. View result = mViewList.get(position);
  68. // 如果mViewList中的view为null,则需要调用newView函数来创建一个,该函数由子类来实现
  69. if (result == null) {
  70. result = newView(position, parent);
  71. mViewList.set(position, result);
  72. }
  73. return result;
  74. }
  75. /**
  76. * 创建列表中指定位置的列表项,该函数有子类实现
  77. */
  78. protected View newView(int position, ViewGroup parent) {
  79. throw new RuntimeException("You must override newView()!");
  80. }
  81. /**
  82. * 获得指定位置的列表项的id
  83. */
  84. @Override
  85. public long getItemId(int position) {
  86. return position;
  87. }
  88. /**
  89. * 判断Adapter中是否存在某个指定的view
  90. */
  91. public boolean hasView(View v) {
  92. return mViewList.contains(v);
  93. }
  94. }

最后,以一个实例来说明SackOfViewAdapter类的使用方法,实例中我们定义4个不同布局的列表项,其中一个为null值,由SackOfViewAdapter的子类重写newView函数实现赋值,其他三个则不是null值。这四个view的布局文件分别是main_notify.xml、update_progress_notify.xml、notification_battery.xml和main_notify_red.xml,这些资源借用自360手机卫士,实例效果如下图所示:

而实例的代码如下:

  1. public class SackOfViewsDemo extends ListActivity {
  2. @Override
  3. public void onCreate(Bundle bundle) {
  4. super.onCreate(bundle);
  5. requestWindowFeature(Window.FEATURE_NO_TITLE);
  6. setContentView(R.layout.main);
  7. ArrayList<View> views = new ArrayList<View>();
  8. LayoutInflater inflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  9. // 列表项1
  10. View view = inflater.inflate(R.layout.main_notify, null);
  11. views.add(view);
  12. // 列表项2
  13. view = inflater.inflate(R.layout.update_progress_notify, null);
  14. views.add(view);
  15. // 列表项3
  16. view = inflater.inflate(R.layout.notification_battery, null);
  17. views.add(view);
  18. // 列表项4(为null值,在newView函数中创建)
  19. views.add(null);
  20. setListAdapter(new SillyAdapter(views));
  21. }
  22. class SillyAdapter extends SackOfViewsAdapter {
  23. public SillyAdapter(List<View> views) {
  24. super(views);
  25. }
  26. protected View newView(int position, ViewGroup parent) {
  27. LayoutInflater inflater = (LayoutInflater) getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  28. View view = inflater.inflate(R.layout.main_notify_red, null);
  29. return view;
  30. }
  31. }
  32. }

本文的SackOfViewAdapter类参考自 https://github.com/commonsguy/cwac-sacklist

本文源码参见:http://download.csdn.net/detail/ace1985/4575749

Android开源代码解读-基于SackOfViewAdapter类实现类似状态通知栏的布局的更多相关文章

  1. Android开源代码解读のOnScrollListener实现ListView滚屏时不加载数据

    使用ListView过程中,如果滚动加载数据的操作比较费时,很容易在滚屏时出现屏幕卡住的现象,一个解决的办法就是不要在滚动时加载数据,而是等到滚动停止后再进行数据的加载.这同样要实现OnScrollL ...

  2. material design 的android开源代码整理

    material design 的android开源代码整理 1 android (material design 效果的代码库) 地址请点击:MaterialDesignLibrary 效果: 2 ...

  3. android开源代码

    Android开源项目--分类汇总 转自:https://github.com/Trinea/android-open-project Android开源项目第一篇——个性化控件(View)篇 包括L ...

  4. 160多个android开源代码汇总

    第一部分 个性化控件(View) 主要介绍那些不错个性化的View,包括ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.Pro ...

  5. 优秀开源代码解读之JS与iOS Native Code互调的优雅实现方案

    简介 本篇为大家介绍一个优秀的开源小项目:WebViewJavascriptBridge. 它优雅地实现了在使用UIWebView时JS与ios 的ObjC nativecode之间的互调,支持消息发 ...

  6. Github上不错的Android开源代码(一)

    总有一些朋友很热心的整理一些好的资料,在收集之后,可以用作阅读.学习和实践.小伙伴们,总有一天,你也能写出 Niubility 的 Android App :-) 为了防止以上链接失效,以及部分内容丢 ...

  7. Hybrid----优秀开源代码解读之JS与iOS Native Code互调的优雅实现方案-备

    本篇为大家介绍一个优秀的开源小项目:WebViewJavascriptBridge. 它优雅地实现了在使用UIWebView时JS与ios 的ObjC nativecode之间的互调,支持消息发送.接 ...

  8. 22个值得收藏的Android开源代码-UI篇

    本文介绍了android开发者中比较热门的开源代码,这些代码绝大多数可以直接应用到项目中. FileBrowserView 一个强大的文件选择控件.界面比较漂亮,使用也很简单.特点:可以自定义UI:支 ...

  9. 22个值得收藏的Android开源代码——cool

    转自http://www.jcodecraeer.com/a/anzhuokaifa/androidkaifa/2014/1020/1808.html 本文介绍了android开发者中比较热门的开源代 ...

随机推荐

  1. 什么是 Terminal

    从用户的角度来看,Terminal 是键盘和显示器的组合,也称为 TTY(电传打字机的缩写).键盘输入字符,显示器显示字符. 从进程的角度来看,终端是字符设备,可以通过 read.write.ioct ...

  2. Swift - defer关键字(推迟执行)

    在一些语言中,有try/finally这样的控制语句,比如Java. 这种语句可以让我们在finally代码块中执行必须要执行的代码,不管之前怎样的兴风作浪. 在Swift 2.0中,Apple提供了 ...

  3. 动态代理写connection连接池Demo

    public class JdbcUtil2 { //声明连接池<放到LinkedList中,操作其中对象的速度快 只需要改变连接> private static LinkedList&l ...

  4. hdu5362 Just A String(dp)

    转载请注明出处: http://www.cnblogs.com/fraud/          ——by fraud Just A String Time Limit: 2000/1000 MS (J ...

  5. [Leetcode][015] 3Sum (Java)

    题目在这里: https://leetcode.com/problems/3sum/ [标签] Array; Two Pointers [个人分析] 老实交待,这个题卡半天,第一次做不会,抄别人的.过 ...

  6. jquery的鼠标移入移出事件hover、mouseenter、mouseleave、mouseover、mouseout

    hover:鼠标进入元素的子元素时不会触发‘鼠标移开’的事件: mouseenter.mouseleave:效果与hover相同: mouseover: 鼠标进入元素和进入它的子元素时都会触发‘mou ...

  7. php基础之三 数组

    一.正则表达式: 1. "/"代表界定符, "^"代表开始符号 "&"结束符号 eg:   $reg="/(13[0-9] ...

  8. Linux的标准输出、标准错误输出、nohup

    1.在bash中标准输出可以用1来表示:通常来说这个1可以省略: 如./xxx >/dev/null 和 ./xxx 1>/dev/null 是一个意思 2.在bash中标准错误输出可以用 ...

  9. TextField控件详解2

    //初始化textfield并设置位置及大小 UITextField *text = [[UITextField alloc]initWithFrame:CGRectMake(20, 20, 130, ...

  10. activiti 5.17 流程图中文乱码问题

    1. 流程图中任务中的中文乱码显示问题.   解决方法:设置processEngineConfiguration中的两个字体属性,例如: <bean id="processEngine ...