Android中ListView控件的使用

 

ListView展示数据的原理

   在Android中,其实ListView就相当于web中的jsp,Adapter是适配器,它就相当于web中的Servlet,

   适配器的作用

    Adapter的作用就是把数据展示在Listview中

使用ListView的奇怪问题?

    在使用ListView的时候,如果把ListView的高设置为wrap_content,它会反复读取多次数据,然后在ListView中把数据显示出来,效率非常低,,这时候我们应该把ListView的高设置为match_parent,这样就能很好的解决读取多次再显示数据的问题了,因为ListView的高写成wrap_content,那么它的高不确定的,需要做多次的校验,确认数据是否能完全显示出来。

  下面我们通过案例说明这个问题

  

安卓listview的item 安卓listview控件的用法_移动开发

  当ListViewf控件的高度设置为wrap_content时,就会出现以下问题,如下图:

 

  

安卓listview的item 安卓listview控件的用法_移动开发_02

        

安卓listview的item 安卓listview控件的用法_java_03

 

               图1                              图2

  我们可以看到手机屏幕图1中最多能够显示31条数据,但是图2中很明显看到当加载完31条记录时,紧接着又从0开始加载这31条记录,其实后面还加载了好几次,在这里就不一一截图出来了,那么如何解决呢?其实只需要修改一下ListView控件的高就可以了,把ListView控件中的高设置为match_parent

  

安卓listview的item 安卓listview控件的用法_java_04

   但是还要注意一点,当是引入布局的时候,我们也需要设置它的父元素的高为match_parent

  

安卓listview的item 安卓listview控件的用法_java_05

  也就是说,父元素和引入布局的ListView都需要设置为match_parent

  

安卓listview的item 安卓listview控件的用法_移动开发_06

  

安卓listview的item 安卓listview控件的用法_安卓listview的item_07

  ListView控件的父子关系关系也是一样

  

安卓listview的item 安卓listview控件的用法_java_08

  

   解决了读取多次数据问题后,我们来看看以下代码,然后运行看看结果是怎样的?


1 import android.app.Activity;
 2 import android.os.Bundle;
 3 import android.view.View;
 4 import android.view.ViewGroup;
 5 import android.widget.BaseAdapter;
 6 import android.widget.ListView;
 7 import android.widget.TextView;
 8 
 9 
10 public class MainActivity extends Activity {
11 
12     @Override
13     protected void onCreate(Bundle savedInstanceState) {
14         super.onCreate(savedInstanceState);
15         setContentView(R.layout.weixin);
16         
17         //获取所需控件
18         ListView ll = (ListView) findViewById(R.id.listView1);
19       
20         //使用适配器
21         ll.setAdapter(new MyAdapter());
22      
23     }
24     
25    //定义一个适配器
26     private class MyAdapter extends BaseAdapter{
27         
28         //返回条目数
29         @Override
30         public int getCount() {
31             return 10000;
32         }
33 
34         @Override
35         public Object getItem(int position) {
36             
37             return null;
38         }
39 
40         @Override
41         public long getItemId(int position) {
42             
43             return 0;
44         }
45         
46         /**
47          * 获取一个view,用来显示listView的数据,会作为listView的一个条目显示
48          * 
49          * position       : 对应getCount()返回的索引
50          * convertView : 缓存数据的对象
51          */
52         @Override
53         public View getView(int position, View convertView, ViewGroup parent) {
54             
55            /**
56              * 如果convertView是null,那么说明没有缓存,那么我们就创建TextView对象
57              */
58             TextView tv = tv = new TextView(MainActivity.this);
59                 System.out.println("创建新的View"+position);
60             
62             tv.setText("呵呵"+position);
63             return tv;
64         }
65         
66     }
67 }


 

  运行结果:

     

安卓listview的item 安卓listview控件的用法_java_09

  我们从结果可以看到,每次都是创建了一个新的对象,这样效率非常低,那么我们下面进行ListView的优化

 

ListView的优化策略

   


1 package com.example.uicustomviews;
 2 
 3 import android.app.Activity;
 4 import android.os.Bundle;
 5 import android.view.View;
 6 import android.view.ViewGroup;
 7 import android.widget.BaseAdapter;
 8 import android.widget.ListView;
 9 import android.widget.TextView;
10 
11 
12 public class MainActivity extends Activity {
13 
14     @Override
15     protected void onCreate(Bundle savedInstanceState) {
16         super.onCreate(savedInstanceState);
17         setContentView(R.layout.weixin);
18         
19         //获取所需控件
20         ListView ll = (ListView) findViewById(R.id.listView1);
21       
22         //使用适配器
23         ll.setAdapter(new MyAdapter());
24      
25     }
26     
27    //定义一个适配器
28     private class MyAdapter extends BaseAdapter{
29         
30         //返回条目数
31         @Override
32         public int getCount() {
33             return 10000;
34         }
35 
36         @Override
37         public Object getItem(int position) {
38             
39             return null;
40         }
41 
42         @Override
43         public long getItemId(int position) {
44             
45             return 0;
46         }
47         
48         /**
49          * 获取一个view,用来显示listView的数据,会作为listView的一个条目显示
50          * 
51          * position       : 对应getCount()返回的索引
52          * convertView : 缓存数据的对象
53          */
54         @Override
55         public View getView(int position, View convertView, ViewGroup parent) {
56             
57             TextView tv = null;
58             
59             /**
60              * 如果convertView是null,那么说明没有缓存,那么我们就创建TextView对象
61              */
62             if(convertView==null){
63                 System.out.println("创建新的View"+position);
64                 tv = new TextView(MainActivity.this);
65             }else{
66                 /**
67                  * 否则就是有缓存,为了提高效率,那么我们就使用缓存中对象,不需要再次new了
68                  */
69                 tv = (TextView) convertView ;
70                 System.out.println("使用缓存的View"+position);
71             }
72             
73             tv.setText("呵呵"+position);
74             return tv;
75         }
76         
77     }
78 }


  运行结果如下图:  

      

安卓listview的item 安卓listview控件的用法_java_10

  显然提高了效率,不再创建新的View,而是使用了缓存中的View

 

 

  下面我们把一个布局文件转为一个View(ListView中的一个条目)


1 package com.example.uicustomviews;
  2 
  3 import android.app.Activity;
  4 import android.os.Bundle;
  5 import android.view.LayoutInflater;
  6 import android.view.View;
  7 import android.view.ViewGroup;
  8 import android.widget.BaseAdapter;
  9 import android.widget.ListView;
 10 
 11 
 12 
 13 public class MainActivity extends Activity {
 14 
 15     @Override
 16     protected void onCreate(Bundle savedInstanceState) {
 17         super.onCreate(savedInstanceState);
 18         setContentView(R.layout.weixin);
 19          
 20         //获取所需控件
 21         ListView ll = (ListView) findViewById(R.id.listView1);
 22       
 23         //使用适配器
 24         ll.setAdapter(new MyAdapter());
 25      
 26     }
 27     
 28    //定义一个适配器
 29     private class MyAdapter extends BaseAdapter{
 30         
 31         //返回条目数
 32         @Override
 33         public int getCount() {
 34             return 10000;
 35         }
 36 
 37         @Override
 38         public Object getItem(int position) {
 39             
 40             return null;
 41         }
 42 
 43         @Override
 44         public long getItemId(int position) {
 45             
 46             return 0;
 47         }
 48         
 49         /**
 50          * 获取一个view,用来显示listView的数据,会作为listView的一个条目显示
 51          * 
 52          * position       : 对应getCount()返回的索引
 53          * convertView : 缓存数据的对象
 54          */
 55         @Override
 56         public View getView(int position, View convertView, ViewGroup parent) {
 57             
 58             /**
 59              * 可以插入广告
 60              */
 61             
 62             View view = null;
 63             
 64             /**
 65              * 如果convertView是null,那么说明没有缓存,那么我们就创建TextView对象
 66              */
 67             if(convertView==null){
 68                 //System.out.println("创建新的View"+position);
 69                 //创建一个新的View对象,可以通过打气筒把一个布局资源转换成一个View对象
 70                 //resource就是我们定义好的布局文件
 71                 //方式一
 72                 //view = View.inflate(MainActivity.this, R.layout.weixin_item, null);
 73                 
 74                 //方式二
 75                 //view = LayoutInflater.from(getApplicationContext()).inflate(R.layout.weixin_item, null);
 76                 
 77                 //方式三
 78                 LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);
 79                 
 80                 view = inflater.inflate(R.layout.weixin_item, null);
 81             }else{
 82                 /**
 83                  * 否则就是有缓存,为了提高效率,那么我们就使用缓存中对象,不需要再次new了
 84                  */
 85                 view = convertView ;
 86                 //System.out.println("使用缓存的View"+position);
 87             }
 88             
 90             return view;
 91         }
 92         
 93     }
 94 }