Android 中 ListView的使用及优化

我们都知道ListView 是在移动端进行显示数据。数据是各种形式的,但是如何让数据清晰化的展示出来,就必须通过ListView 这样的控件来实现。在Android 机制中,在数据源和ListView中间还存在一个适配器Adapter。

将数据适配到ListView中,我们首先是将数据已经存入到列表中,然后才将数据一条一条适配。其中,adapter 中最核心的方法便是
public View getView(int position, View convertView, ViewGroup parent)
该方法内包含着Android 系统的缓存机制。

代码一

假设我们是Android 系统知识的小白,不懂得什么ListView 的优化,我们可能第一次磕磕绊绊写出来的代码是这样的。这里只贴出adapter的代码。

@Override
    public View getView(int position, View convertView, ViewGroup parent) {
        Fruit fruit = getItem(position); // 获取当前项的Fruit实例
        View view = LayoutInflater.from(getContext()).inflate(resourceId, null);
        ImageView fruitImage = (ImageView) view.findViewById(R.id.fruit_image);
        TextView fruitName = (TextView) view.findViewById(R.id.fruit_name);
        fruitImage.setImageResource(fruit.getImageId());
        fruitName.setText(fruit.getName());
        return view;
    }

我们假设我们实现的列表中有5条数据,那么这个数据加载过程是看不出来差异的。如果列表数据量非常大,假设有成千上万条数据,并且上下滑动速度飞快,那么可能出现数据加载不出来的情况,怎么解决呢?分析下我们的代码。

当每条数据进入可见区时,adapter的getView() 会被调用,返回代表具体数据的视图;触摸滚动时,频繁调用;支持成百上千条数据。

代码二

position位置不同,得到的fruit 实例不同,但每次执行到这个方法时总会重新获取自定义View 的布局。那么我们能不能对其进行限制,减少通过这种方式来获取布局。

@Override
    public View getView(int position, View convertView, ViewGroup parent) {
        Fruit fruit = getItem(position); // 获取当前项的Fruit实例
        if(convertView == null)
        {
            convertView = LayoutInflater.from(getContext()).inflate(resourceId, null);
            //convertView = view;
            Log.i("tag", "positon " + position + " convertView is null, " + "new: " + convertView);  
        }else
        {
            Log.i("tag", "positon " + position + " convertView is not null, " + "new: " + convertView);  
        }
        ImageView fruitImage = (ImageView) convertView.findViewById(R.id.fruit_image);
        TextView fruitName = (TextView) convertView.findViewById(R.id.fruit_name);
        fruitImage.setImageResource(fruit.getImageId());
        fruitName.setText(fruit.getName());
        return convertView;
    }

这样利用Android 的缓存机制对ListView 进行了优化,回收效率提高了200%(怎么计算的?还在思考中……自己认为这个应该和自己手机屏幕每次加载的数据条数有关吧?有待考证)为什么会存在这种优化?我查看了先辈的博客,其中有几篇很好,如
http://www.cnblogs.com/xiaowenji/archive/2010/12/08/1900579.htmlhttp://haking.iteye.com/blog/1147404这两篇都对其机制有一个大概的陈述,按照他们的步骤,是可以得到结论的。

代码三

再看看还有没有需要优化的地方……
现在这个view布局已经存在,每个Item中的数据需要适配。当每个Item的布局一样时,就只是数据有所不同,显而易见,我们不需要每次都去找到那个控件,可以使用对象进行保存。于是ViewHolder出现了。代码如下:

@Override
    public View getView(int position, View convertView, ViewGroup parent) {

        Fruit fruit = getItem(position);
        View view;
        ViewHolder viewHolder;
        if (convertView == null) {
            view = LayoutInflater.from(getContext()).inflate(resourceId, null);
            viewHolder = new ViewHolder();
            viewHolder.fruitImage = (ImageView) view.findViewById(R.id.fruit_image);
            viewHolder.fruitName = (TextView) view.findViewById(R.id.fruit_name);
            view.setTag(viewHolder);
            Log.i("tag", "positon " + position + " convertView is null, " + "new: " + convertView);  
        } else {
            view = convertView;
            viewHolder = (ViewHolder) view.getTag();
             Log.i("tag", "positon " + position + " convertView is not null, "  + convertView);  
        }
        viewHolder.fruitImage.setImageResource(fruit.getImageId());
        viewHolder.fruitName.setText(fruit.getName());
        return view;
    }

    class ViewHolder {

        ImageView fruitImage;

        TextView fruitName;

    }

在这中间,产生了一个类ViewHolder,产生对象进行保存数据fruitImage,fruitName。当一个View 为空时,就使用setTag() 方法为View 添加数据,否则就取出数据 。这样也进行了优化。
至此,将listview 的优化过程已经写完了,欢迎指正切磋!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值