Android中ListView控件的使用

时间:2023-03-08 16:39:09
Android中ListView控件的使用

Android中ListView控件的使用

ListView展示数据的原理

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

  适配器的作用

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

使用ListView的奇怪问题?

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

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

  Android中ListView控件的使用

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

  Android中ListView控件的使用        Android中ListView控件的使用

               图1                              图2

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

  Android中ListView控件的使用

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

  Android中ListView控件的使用

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

  Android中ListView控件的使用  Android中ListView控件的使用

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

  Android中ListView控件的使用

  

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

 import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView; public class MainActivity extends Activity { @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.weixin); //获取所需控件
ListView ll = (ListView) findViewById(R.id.listView1); //使用适配器
ll.setAdapter(new MyAdapter()); } //定义一个适配器
private class MyAdapter extends BaseAdapter{ //返回条目数
@Override
public int getCount() {
return 10000;
} @Override
public Object getItem(int position) { return null;
} @Override
public long getItemId(int position) { return 0;
} /**
* 获取一个view,用来显示listView的数据,会作为listView的一个条目显示
*
* position : 对应getCount()返回的索引
* convertView : 缓存数据的对象
*/
@Override
public View getView(int position, View convertView, ViewGroup parent) { /**
* 如果convertView是null,那么说明没有缓存,那么我们就创建TextView对象
*/
TextView tv = tv = new TextView(MainActivity.this);
System.out.println("创建新的View"+position); tv.setText("呵呵"+position);
return tv;
} }
}

  运行结果:

     Android中ListView控件的使用

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

ListView的优化策略

   

 package com.example.uicustomviews;

 import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ListView;
import android.widget.TextView; public class MainActivity extends Activity { @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.weixin); //获取所需控件
ListView ll = (ListView) findViewById(R.id.listView1); //使用适配器
ll.setAdapter(new MyAdapter()); } //定义一个适配器
private class MyAdapter extends BaseAdapter{ //返回条目数
@Override
public int getCount() {
return 10000;
} @Override
public Object getItem(int position) { return null;
} @Override
public long getItemId(int position) { return 0;
} /**
* 获取一个view,用来显示listView的数据,会作为listView的一个条目显示
*
* position : 对应getCount()返回的索引
* convertView : 缓存数据的对象
*/
@Override
public View getView(int position, View convertView, ViewGroup parent) { TextView tv = null; /**
* 如果convertView是null,那么说明没有缓存,那么我们就创建TextView对象
*/
if(convertView==null){
System.out.println("创建新的View"+position);
tv = new TextView(MainActivity.this);
}else{
/**
* 否则就是有缓存,为了提高效率,那么我们就使用缓存中对象,不需要再次new了
*/
tv = (TextView) convertView ;
System.out.println("使用缓存的View"+position);
} tv.setText("呵呵"+position);
return tv;
} }
}

  运行结果如下图:  

      Android中ListView控件的使用

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

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

 package com.example.uicustomviews;

 import android.app.Activity;
import android.os.Bundle;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.ListView; public class MainActivity extends Activity { @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.weixin); //获取所需控件
ListView ll = (ListView) findViewById(R.id.listView1); //使用适配器
ll.setAdapter(new MyAdapter()); } //定义一个适配器
private class MyAdapter extends BaseAdapter{ //返回条目数
@Override
public int getCount() {
return 10000;
} @Override
public Object getItem(int position) { return null;
} @Override
public long getItemId(int position) { return 0;
} /**
* 获取一个view,用来显示listView的数据,会作为listView的一个条目显示
*
* position : 对应getCount()返回的索引
* convertView : 缓存数据的对象
*/
@Override
public View getView(int position, View convertView, ViewGroup parent) { /**
* 可以插入广告
*/ View view = null; /**
* 如果convertView是null,那么说明没有缓存,那么我们就创建TextView对象
*/
if(convertView==null){
//System.out.println("创建新的View"+position);
//创建一个新的View对象,可以通过打气筒把一个布局资源转换成一个View对象
70 //resource就是我们定义好的布局文件
//方式一
//view = View.inflate(MainActivity.this, R.layout.weixin_item, null); //方式二
//view = LayoutInflater.from(getApplicationContext()).inflate(R.layout.weixin_item, null); //方式三
LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE); view = inflater.inflate(R.layout.weixin_item, null);
}else{
/**
* 否则就是有缓存,为了提高效率,那么我们就使用缓存中对象,不需要再次new了
*/
view = convertView ;
//System.out.println("使用缓存的View"+position);
} return view;
} }
}