加载中...

2.5.0 构建一个可复用的自定义BaseAdapter


本节引言:

如题,本节给大家带来的是构建一个可复用的自定义BaseAdapter,我们每每涉及到ListView GridView等其他的Adapter控件,都需要自己另外写一个BaseAdapter类,这样显得非常麻烦, 又比如,我们想在一个界面显示两个ListView的话,我们也是需要些两个BaseAdapter... 这,程序员都是喜欢偷懒的哈,这节我们就来写一个可复用的自定义BaseAdapter类~

1.我们一点点开始改:

首先我们把上节写的自定义BaseAdapter贴下,等下我们就要对他进行升级改造

  1. /**
  2. * Created by Jay on 2015/9/21 0021.
  3. */
  4. public class MyAdapter extends BaseAdapter {
  5.  
  6. private Context mContext;
  7. private LinkedList<Data> mData;
  8.  
  9. public MyAdapter() {
  10. }
  11.  
  12. public MyAdapter(LinkedList<Data> mData, Context mContext) {
  13. this.mData = mData;
  14. this.mContext = mContext;
  15. }
  16.  
  17. @Override
  18. public int getCount() {
  19. return mData.size();
  20. }
  21.  
  22. @Override
  23. public Object getItem(int position) {
  24. return null;
  25. }
  26.  
  27. @Override
  28. public long getItemId(int position) {
  29. return position;
  30. }
  31.  
  32. @Override
  33. public View getView(int position, View convertView, ViewGroup parent) {
  34. ViewHolder holder = null;
  35. if (convertView == null) {
  36. convertView = LayoutInflater.from(mContext).inflate(R.layout.item_list, parent, false);
  37. holder = new ViewHolder();
  38. holder.img_icon = (ImageView) convertView.findViewById(R.id.img_icon);
  39. holder.txt_content = (TextView) convertView.findViewById(R.id.txt_content);
  40. convertView.setTag(holder);
  41. } else {
  42. holder = (ViewHolder) convertView.getTag();
  43. }
  44. holder.img_icon.setImageResource(mData.get(position).getImgId());
  45. holder.txt_content.setText(mData.get(position).getContent());
  46. return convertView;
  47. }
  48.  
  49. //添加一个元素
  50. public void add(Data data) {
  51. if (mData == null) {
  52. mData = new LinkedList<>();
  53. }
  54. mData.add(data);
  55. notifyDataSetChanged();
  56. }
  57.  
  58. //往特定位置,添加一个元素
  59. public void add(int position,Data data){
  60. if (mData == null) {
  61. mData = new LinkedList<>();
  62. }
  63. mData.add(position, data);
  64. notifyDataSetChanged();
  65. }
  66.  
  67. public void remove(Data data) {
  68. if(mData != null) {
  69. mData.remove(data);
  70. }
  71. notifyDataSetChanged();
  72. }
  73.  
  74. public void remove(int position) {
  75. if(mData != null) {
  76. mData.remove(position);
  77. }
  78. notifyDataSetChanged();
  79. }
  80.  
  81. public void clear() {
  82. if(mData != null) {
  83. mData.clear();
  84. }
  85. notifyDataSetChanged();
  86. }
  87.  
  88. private class ViewHolder {
  89. ImageView img_icon;
  90. TextView txt_content;
  91. }
  92.  
  93. }

升级1:将Entity设置成泛型

好的,毕竟我们传递过来的Entitiy实体类可能千奇百怪,比如有Person,Book,Wether等,所以我们 将Entity设置成泛型,修改后的代码如下:

  1. <pre>
  2. public class MyAdapter<T> extends BaseAdapter {
  3.  
  4. private Context mContext;
  5. private LinkedList<T> mData;
  6.  
  7. public MyAdapter() {
  8. }
  9.  
  10. public MyAdapter(LinkedList<T> mData, Context mContext) {
  11. this.mData = mData;
  12. this.mContext = mContext;
  13. }
  14.  
  15. @Override
  16. public int getCount() {
  17. return mData.size();
  18. }
  19.  
  20. @Override
  21. public Object getItem(int position) {
  22. return null;
  23. }
  24.  
  25. @Override
  26. public long getItemId(int position) {
  27. return position;
  28. }
  29.  
  30. @Override
  31. public View getView(int position, View convertView, ViewGroup parent) {
  32. ViewHolder holder = null;
  33. if (convertView == null) {
  34. convertView = LayoutInflater.from(mContext).inflate(R.layout.item_list, parent, false);
  35. holder = new ViewHolder();
  36. holder.img_icon = (ImageView) convertView.findViewById(R.id.img_icon);
  37. holder.txt_content = (TextView) convertView.findViewById(R.id.txt_content);
  38. convertView.setTag(holder);
  39. } else {
  40. holder = (ViewHolder) convertView.getTag();
  41. }
  42. holder.img_icon.setImageResource(mData.get(position).getImgId());
  43. holder.txt_content.setText(mData.get(position).getContent());
  44. return convertView;
  45. }
  46.  
  47. //添加一个元素
  48. public void add(T data) {
  49. if (mData == null) {
  50. mData = new LinkedList<>();
  51. }
  52. mData.add(data);
  53. notifyDataSetChanged();
  54. }
  55.  
  56. //往特定位置,添加一个元素
  57. public void add(int position,T data){
  58. if (mData == null) {
  59. mData = new LinkedList<>();
  60. }
  61. mData.add(position, data);
  62. notifyDataSetChanged();
  63. }
  64.  
  65. public void remove(T data) {
  66. if(mData != null) {
  67. mData.remove(data);
  68. }
  69. notifyDataSetChanged();
  70. }
  71.  
  72. public void remove(int position) {
  73. if(mData != null) {
  74. mData.remove(position);
  75. }
  76. notifyDataSetChanged();
  77. }
  78.  
  79. public void clear() {
  80. if(mData != null) {
  81. mData.clear();
  82. }
  83. notifyDataSetChanged();
  84. }
  85.  
  86. private class ViewHolder {
  87. ImageView img_icon;
  88. TextView txt_content;
  89. }
  90.  
  91. }

好的,上面我们做的事仅仅是将Data类型换成了泛型T!

升级2:ViewHolder类的升级改造:

我们先来看看前面我们的ViewHolder干了什么? 答:findViewById,设置控件状态; 下面我们想在完成这个基础上,将getView()方法大部分的逻辑写到ViewHolder类里, 这个ViewHolder要做的事:

  • 定义一个查找控件的方法,我们的思路是通过暴露公共的方法,调用方法时传递过来 控件id,以及设置的内容,比如TextView设置文本: public ViewHolder setText(int id, CharSequence text){文本设置}
  • 将convertView复用部分搬到这里,那就需要传递一个context对象了,我们把需要获取 的部分都写到构造方法中!
  • 写一堆设置方法(public),比如设置文字大小颜色,图片背景等!

好的,接下来我们就来一步步改造我们的ViewHolder类

1)相关参数与构造方法:

  1. public static class ViewHolder {
  2.  
  3. private SparseArray<View> mViews; //存储ListView 的 item中的View
  4. private View item; //存放convertView
  5. private int position; //游标
  6. private Context context; //Context上下文
  7.  
  8. //构造方法,完成相关初始化
  9. private ViewHolder(Context context, ViewGroup parent, int layoutRes) {
  10. mViews = new SparseArray<>();
  11. this.context = context;
  12. View convertView = LayoutInflater.from(context).inflate(layoutRes, parent,false);
  13. convertView.setTag(this);
  14. item = convertView;
  15. }
  16.  
  17. ImageView img_icon;
  18. TextView txt_content;
  19. }

2)绑定ViewHolder与Item

在上面的基础上我们再添加一个绑定的方法

  1. //绑定ViewHolder与item
  2. public static ViewHolder bind(Context context, View convertView, ViewGroup parent,
  3. int layoutRes, int position) {
  4. ViewHolder holder;
  5. if(convertView == null) {
  6. holder = new ViewHolder(context, parent, layoutRes);
  7. } else {
  8. holder = (ViewHolder) convertView.getTag();
  9. holder.item = convertView;
  10. }
  11. holder.position = position;
  12. return holder;
  13. }

3)根据id获取集合中保存的控件

  1. public <T extends View> T getView(int id) {
  2. T t = (T) mViews.get(id);
  3. if(t == null) {
  4. t = (T) item.findViewById(id);
  5. mViews.put(id, t);
  6. }
  7. return t;
  8. }

4) 接着我们再定义一堆暴露出来的方法

  1. /**
  2. * 获取当前条目
  3. */
  4. public View getItemView() {
  5. return item;
  6. }
  7.  
  8. /**
  9. * 获取条目位置
  10. */
  11. public int getItemPosition() {
  12. return position;
  13. }
  14.  
  15. /**
  16. * 设置文字
  17. */
  18. public ViewHolder setText(int id, CharSequence text) {
  19. View view = getView(id);
  20. if(view instanceof TextView) {
  21. ((TextView) view).setText(text);
  22. }
  23. return this;
  24. }
  25.  
  26. /**
  27. * 设置图片
  28. */
  29. public ViewHolder setImageResource(int id, int drawableRes) {
  30. View view = getView(id);
  31. if(view instanceof ImageView) {
  32. ((ImageView) view).setImageResource(drawableRes);
  33. } else {
  34. view.setBackgroundResource(drawableRes);
  35. }
  36. return this;
  37. }
  38.  
  39.  
  40. /**
  41. * 设置点击监听
  42. */
  43. public ViewHolder setOnClickListener(int id, View.OnClickListener listener) {
  44. getView(id).setOnClickListener(listener);
  45. return this;
  46. }
  47.  
  48. /**
  49. * 设置可见
  50. */
  51. public ViewHolder setVisibility(int id, int visible) {
  52. getView(id).setVisibility(visible);
  53. return this;
  54. }
  55.  
  56. /**
  57. * 设置标签
  58. */
  59. public ViewHolder setTag(int id, Object obj) {
  60. getView(id).setTag(obj);
  61. return this;
  62. }
  63.  
  64. //其他方法可自行扩展

好的,ViewHolder的改造升级完成~

升级3:定义一个抽象方法,完成ViewHolder与Data数据集的绑定

  1. public abstract void bindView(ViewHolder holder, T obj);

我们创建新的BaseAdapter的时候,实现这个方法就好,另外,别忘了把我们自定义 的BaseAdapter改成abstact抽象的!

升级4:修改getView()部分的内容

  1. @Override
  2. public View getView(int position, View convertView, ViewGroup parent) {
  3. ViewHolder holder = ViewHolder.bind(parent.getContext(), convertView, parent, mLayoutRes
  4. , position);
  5. bindView(holder,getItem(position));
  6. return holder.getItemView();
  7. }

2.升级完毕,我们写代码来体验下:

我们要实现的效果图:

就是上面有两个列表,布局不一样,但是我只使用一个BaseAdapter类来完成上述效果!

关键代码如下:

MainActivity.java

  1. public class MainActivity extends AppCompatActivity {
  2.  
  3. private Context mContext;
  4. private ListView list_book;
  5. private ListView list_app;
  6.  
  7. private MyAdapter<App> myAdapter1 = null;
  8. private MyAdapter<Book> myAdapter2 = null;
  9. private List<App> mData1 = null;
  10. private List<Book> mData2 = null;
  11.  
  12. @Override
  13. protected void onCreate(Bundle savedInstanceState) {
  14. super.onCreate(savedInstanceState);
  15. setContentView(R.layout.activity_main);
  16. mContext = MainActivity.this;
  17. init();
  18.  
  19. }
  20.  
  21. private void init() {
  22.  
  23. list_book = (ListView) findViewById(R.id.list_book);
  24. list_app = (ListView) findViewById(R.id.list_app);
  25.  
  26. //数据初始化
  27. mData1 = new ArrayList<App>();
  28. mData1.add(new App(R.mipmap.iv_icon_baidu,"百度"));
  29. mData1.add(new App(R.mipmap.iv_icon_douban,"豆瓣"));
  30. mData1.add(new App(R.mipmap.iv_icon_zhifubao,"支付宝"));
  31.  
  32. mData2 = new ArrayList<Book>();
  33. mData2.add(new Book("《第一行代码Android》","郭霖"));
  34. mData2.add(new Book("《Android群英传》","徐宜生"));
  35. mData2.add(new Book("《Android开发艺术探索》","任玉刚"));
  36.  
  37. //Adapter初始化
  38. myAdapter1 = new MyAdapter<App>((ArrayList)mData1,R.layout.item_one) {
  39. @Override
  40. public void bindView(ViewHolder holder, App obj) {
  41. holder.setImageResource(R.id.img_icon,obj.getaIcon());
  42. holder.setText(R.id.txt_aname,obj.getaName());
  43. }
  44. };
  45. myAdapter2 = new MyAdapter<Book>((ArrayList)mData2,R.layout.item_two) {
  46. @Override
  47. public void bindView(ViewHolder holder, Book obj) {
  48. holder.setText(R.id.txt_bname,obj.getbName());
  49. holder.setText(R.id.txt_bauthor,obj.getbAuthor());
  50. }
  51. };
  52.  
  53. //ListView设置下Adapter:
  54. list_book.setAdapter(myAdapter2);
  55. list_app.setAdapter(myAdapter1);
  56.  
  57. }
  58.  
  59.  
  60. }

我们写的可复用的BaseAdapter的使用就如上面所述~

3.代码示例下载:

ListViewDemo4.zip

贴下最后写好的MyAdapter类吧,可根据自己的需求进行扩展:

MyAdapter.java

  1. /**
  2. * Created by Jay on 2015/9/22 0022.
  3. */
  4. public abstract class MyAdapter<T> extends BaseAdapter {
  5.  
  6. private ArrayList<T> mData;
  7. private int mLayoutRes; //布局id
  8.  
  9.  
  10. public MyAdapter() {
  11. }
  12.  
  13. public MyAdapter(ArrayList<T> mData, int mLayoutRes) {
  14. this.mData = mData;
  15. this.mLayoutRes = mLayoutRes;
  16. }
  17.  
  18. @Override
  19. public int getCount() {
  20. return mData != null ? mData.size() : 0;
  21. }
  22.  
  23. @Override
  24. public T getItem(int position) {
  25. return mData.get(position);
  26. }
  27.  
  28. @Override
  29. public long getItemId(int position) {
  30. return position;
  31. }
  32.  
  33. @Override
  34. public View getView(int position, View convertView, ViewGroup parent) {
  35. ViewHolder holder = ViewHolder.bind(parent.getContext(), convertView, parent, mLayoutRes
  36. , position);
  37. bindView(holder, getItem(position));
  38. return holder.getItemView();
  39. }
  40.  
  41. public abstract void bindView(ViewHolder holder, T obj);
  42.  
  43. //添加一个元素
  44. public void add(T data) {
  45. if (mData == null) {
  46. mData = new ArrayList<>();
  47. }
  48. mData.add(data);
  49. notifyDataSetChanged();
  50. }
  51.  
  52. //往特定位置,添加一个元素
  53. public void add(int position, T data) {
  54. if (mData == null) {
  55. mData = new ArrayList<>();
  56. }
  57. mData.add(position, data);
  58. notifyDataSetChanged();
  59. }
  60.  
  61. public void remove(T data) {
  62. if (mData != null) {
  63. mData.remove(data);
  64. }
  65. notifyDataSetChanged();
  66. }
  67.  
  68. public void remove(int position) {
  69. if (mData != null) {
  70. mData.remove(position);
  71. }
  72. notifyDataSetChanged();
  73. }
  74.  
  75. public void clear() {
  76. if (mData != null) {
  77. mData.clear();
  78. }
  79. notifyDataSetChanged();
  80. }
  81.  
  82.  
  83. public static class ViewHolder {
  84.  
  85. private SparseArray<View> mViews; //存储ListView 的 item中的View
  86. private View item; //存放convertView
  87. private int position; //游标
  88. private Context context; //Context上下文
  89.  
  90. //构造方法,完成相关初始化
  91. private ViewHolder(Context context, ViewGroup parent, int layoutRes) {
  92. mViews = new SparseArray<>();
  93. this.context = context;
  94. View convertView = LayoutInflater.from(context).inflate(layoutRes, parent, false);
  95. convertView.setTag(this);
  96. item = convertView;
  97. }
  98.  
  99. //绑定ViewHolder与item
  100. public static ViewHolder bind(Context context, View convertView, ViewGroup parent,
  101. int layoutRes, int position) {
  102. ViewHolder holder;
  103. if (convertView == null) {
  104. holder = new ViewHolder(context, parent, layoutRes);
  105. } else {
  106. holder = (ViewHolder) convertView.getTag();
  107. holder.item = convertView;
  108. }
  109. holder.position = position;
  110. return holder;
  111. }
  112.  
  113. @SuppressWarnings("unchecked")
  114. public <T extends View> T getView(int id) {
  115. T t = (T) mViews.get(id);
  116. if (t == null) {
  117. t = (T) item.findViewById(id);
  118. mViews.put(id, t);
  119. }
  120. return t;
  121. }
  122.  
  123.  
  124. /**
  125. * 获取当前条目
  126. */
  127. public View getItemView() {
  128. return item;
  129. }
  130.  
  131. /**
  132. * 获取条目位置
  133. */
  134. public int getItemPosition() {
  135. return position;
  136. }
  137.  
  138. /**
  139. * 设置文字
  140. */
  141. public ViewHolder setText(int id, CharSequence text) {
  142. View view = getView(id);
  143. if (view instanceof TextView) {
  144. ((TextView) view).setText(text);
  145. }
  146. return this;
  147. }
  148.  
  149. /**
  150. * 设置图片
  151. */
  152. public ViewHolder setImageResource(int id, int drawableRes) {
  153. View view = getView(id);
  154. if (view instanceof ImageView) {
  155. ((ImageView) view).setImageResource(drawableRes);
  156. } else {
  157. view.setBackgroundResource(drawableRes);
  158. }
  159. return this;
  160. }
  161.  
  162.  
  163. /**
  164. * 设置点击监听
  165. */
  166. public ViewHolder setOnClickListener(int id, View.OnClickListener listener) {
  167. getView(id).setOnClickListener(listener);
  168. return this;
  169. }
  170.  
  171. /**
  172. * 设置可见
  173. */
  174. public ViewHolder setVisibility(int id, int visible) {
  175. getView(id).setVisibility(visible);
  176. return this;
  177. }
  178.  
  179. /**
  180. * 设置标签
  181. */
  182. public ViewHolder setTag(int id, Object obj) {
  183. getView(id).setTag(obj);
  184. return this;
  185. }
  186.  
  187. //其他方法可自行扩展
  188.  
  189. }
  190.  
  191. }

本节小结:

本节给大家介绍了如何来实现一个可供复用的BaseAdapter,当然大家可以在这个的基础上根据 自己的需求进行修改,比如通过异步设置网络图片等~改代码是参考鸿洋大神的视频写的: 视频链接:Android-打造万能适配器 另外,实际编写中遇到一些问题,非常感谢Berial(B神)的耐心点拨~ ありがとうございます~


还没有评论.