zoukankan      html  css  js  c++  java
  • android平板上的GridView视图缓存优化

    最近在做android平板上的开发,其中涉及到高分辨率之下使用GridView的性能问题。在Android手机软件开发中,如果在ListView或者GridView上使用大数量Item,很多人都会想到ViewHolder......没错,ViewHolder非常适合用在ListView或者每行小于4个Item的GridView。但是如果是高分辨率的设备(android平板甚至android电视),每行包含4个以上Item的话,即使用了ViewHolder也依然卡。

          如下图,每行9个Item,而且每个Item的图片都是从网络动态下载的,这时就比较考验GridView视图的优化了。

          本文提出的优化方法是:在getView()构建一个View列表(List<View>),把最近构建的View存起来,回退时直接从View列表中读取,而不是动态构建。使用这种方法有2个好处:

    1.快速读取过去的Item;

    2.直接保存View而不是Bitmap,避免了ImageView.setImageBitmaps()带来的延时。

    当然坏处就是浪费内存,所以要设定一个上限,超过了就删掉最老的Item。
    先来看看这种方法与ViewHolder的性能对比:


    100个Item往下滚到的三组数据对比,如上图:
    “CacheAdapter 缓存50个Item”跟ViewHolderAdapter的速度很接近,由于CacheAdapter有缓存,所以会有1~2次快速读取Item(10~20个)的情况,而ViewHolder的每次读取Item速度比较平均。
    “CacheAdapter 缓存75个Item”只在第一次往下滚动时消耗较长时间,第二次用了缓存的Item,所以速度快了很多。

    100个Item往上滚到的三组数据对比,如上图:

    “CacheAdapter 缓存50个Item”比ViewHolderAdapter的速度略快,“CacheAdapter 缓存75个Item”依然是最快的。
    总结:“CacheAdapter 缓存50个Item”速度与HolderView略快,读取最近的Item速度最快,缓存的Item越多速度越快。“CacheAdapter 缓存75个Item”占用内存最少,这是由于一部分图片下载失败,保存的Item的图片为空,实际上是缓存越多Item占用的内存越多。

    PS:这里用到异步读取网络图片,成功下载的就占用较多内存,下载失败就占用较少内存,所以内存占用情况并不是一个时刻的绝对值,占用内存只用于参考.....

    本文程序源码可以到http://www.rayfile.com/zh-cn/files/5ebf5666-958a-11e0-99ec-0015c55db73d/这里下载。

    CacheAdapter.java是实现缓存Item的自定义Adapter,源码如下:

    /**
    * 使用列表缓存过去的Item
    *
    @author hellogv
    *
    */
    public class CacheAdapter extends BaseAdapter {

    public class Item {
    public String itemImageURL;
    public String itemTitle;
    public Item(String itemImageURL, String itemTitle) {
    this.itemImageURL = itemImageURL;
    this.itemTitle = itemTitle;
    }
    }

    private Context mContext;
    private ArrayList<Item> mItems = new ArrayList<Item>();
    LayoutInflater inflater;
    public CacheAdapter(Context c) {
    mContext = c;
    inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    }

    public void addItem(String itemImageURL, String itemTitle) {
    mItems.add(new Item(itemImageURL, itemTitle));
    }

    public int getCount() {
    return mItems.size();
    }

    public Item getItem(int position) {
    return mItems.get(position);
    }

    public long getItemId(int position) {
    return position;
    }

    List<Integer> lstPosition=new ArrayList<Integer>();
    List<View> lstView=new ArrayList<View>();

    List<Integer> lstTimes= new ArrayList<Integer>();
    long startTime=0;
    public View getView(int position, View convertView, ViewGroup parent) {
    startTime=System.nanoTime();

    if (lstPosition.contains(position) == false) {
    if(lstPosition.size()>75)//这里设置缓存的Item数量
    {
    lstPosition.remove(0);//删除第一项
    lstView.remove(0);//删除第一项
    }
    convertView = inflater.inflate(R.layout.item, null);
    TextView text = (TextView) convertView.findViewById(R.id.itemText);
    ImageView icon = (ImageView) convertView.findViewById(R.id.itemImage);
    text.setText(mItems.get(position).itemTitle);
    new AsyncLoadImage().execute(new Object[] { icon,mItems.get(position).itemImageURL });

    lstPosition.add(position);//添加最新项
    lstView.add(convertView);//添加最新项
    } else
    {
    convertView = lstView.get(lstPosition.indexOf(position));
    }

    int endTime=(int) (System.nanoTime()-startTime);
    lstTimes.add(endTime);
    if(lstTimes.size()==10)
    {
    int total=0;
    for(int i=0;i<lstTimes.size();i++)
    total=total+lstTimes.get(i);

    Log.e("10个所花的时间:" +total/1000 +" μs",
    "所用内存:"+Runtime.getRuntime().totalMemory()/1024 +" KB");
    lstTimes.clear();
    }

    return convertView;
    }

    /**
    * 异步读取网络图片
    *
    @author hellogv
    */
    class AsyncLoadImage extends AsyncTask<Object, Object, Void> {
    @Override
    protected Void doInBackground(Object... params) {

    try {
    ImageView imageView=(ImageView) params[0];
    String url=(String) params[1];
    Bitmap bitmap = getBitmapByUrl(url);
    publishProgress(new Object[] {imageView, bitmap});
    } catch (MalformedURLException e) {
    Log.e("error",e.getMessage());
    e.printStackTrace();
    } catch (IOException e) {
    Log.e("error",e.getMessage());
    e.printStackTrace();
    }
    return null;
    }

    protected void onProgressUpdate(Object... progress) {
    ImageView imageView = (ImageView) progress[0];
    imageView.setImageBitmap((Bitmap) progress[1]);
    }
    }

    static public Bitmap getBitmapByUrl(String urlString)
    throws MalformedURLException, IOException {
    URL url = new URL(urlString);
    URLConnection connection = url.openConnection();
    connection.setConnectTimeout(25000);
    connection.setReadTimeout(90000);
    Bitmap bitmap = BitmapFactory.decodeStream(connection.getInputStream());
    return bitmap;
    }
    }

    其中if(lstPosition.size()>75)是设置缓存的Item数量的关键地方,这里缓存75个Item。

    ViewHolderAdapter.java是实现ViewHolder加载Item的自定义Adapter,源码如下:

    /**
    * 使用ViewHolder加载Item
    *
    @author hellogv
    *
    */
    public class ViewHolderAdapter extends BaseAdapter {

    public class Item {
    public String itemImageURL;
    public String itemTitle;

    public Item(String itemImageURL, String itemTitle) {
    this.itemImageURL = itemImageURL;
    this.itemTitle = itemTitle;
    }
    }

    private Context mContext;
    private ArrayList<Item> mItems = new ArrayList<Item>();
    LayoutInflater inflater;
    public ViewHolderAdapter(Context c) {
    mContext = c;
    inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    }

    public void addItem(String itemImageURL, String itemTitle) {
    mItems.add(new Item(itemImageURL, itemTitle));
    }

    public int getCount() {
    return mItems.size();
    }

    public Item getItem(int position) {
    return mItems.get(position);
    }

    public long getItemId(int position) {
    return position;
    }

    static class ViewHolder {
    TextView text;
    ImageView icon;
    }

    List<Integer> lstTimes= new ArrayList<Integer>();
    long startTime=0;
    public View getView(int position, View convertView, ViewGroup parent) {
    startTime=System.nanoTime();

    ViewHolder holder;

    if (convertView == null) {
    convertView = inflater.inflate(R.layout.item, null);
    holder = new ViewHolder();
    holder.text = (TextView) convertView.findViewById(R.id.itemText);
    holder.icon = (ImageView) convertView.findViewById(R.id.itemImage);
    convertView.setTag(holder);
    } else {
    holder = (ViewHolder) convertView.getTag();
    }
    holder.text.setText(mItems.get(position).itemTitle);
    new AsyncLoadImage().execute(new Object[]{holder.icon,mItems.get(position).itemImageURL });

    int endTime=(int) (System.nanoTime()-startTime);
    lstTimes.add(endTime);
    if(lstTimes.size()==10)
    {
    int total=0;
    for(int i=0;i<lstTimes.size();i++)
    total=total+lstTimes.get(i);

    Log.e("10个所花的时间:" +total/1000 +" μs",
    "所用内存:"+Runtime.getRuntime().totalMemory()/1024 +" KB");
    lstTimes.clear();
    }

    return convertView;
    }

    /**
    * 异步读取网络图片
    *
    @author hellogv
    */
    class AsyncLoadImage extends AsyncTask<Object, Object, Void> {
    @Override
    protected Void doInBackground(Object... params) {

    try {
    ImageView imageView=(ImageView) params[0];
    String url=(String) params[1];
    Bitmap bitmap = CacheAdapter.getBitmapByUrl(url);
    publishProgress(new Object[] {imageView, bitmap});
    } catch (MalformedURLException e) {
    e.printStackTrace();
    } catch (IOException e) {
    e.printStackTrace();
    }
    return null;
    }

    protected void onProgressUpdate(Object... progress) {
    ImageView imageView = (ImageView) progress[0];
    imageView.setImageBitmap((Bitmap) progress[1]);
    }
    }

    }

    testPerformance.java是主程序,通过注释符就可以分别测试CacheAdapter与ViewHolderAdapter的性能,源码如下:

    public class testPerformance extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    this.setTitle("android平板上的GridView视图缓存优化-----hellogv");
    GridView gridview = (GridView) findViewById(R.id.gridview);
    CacheAdapter adapter=new CacheAdapter(this);
    // ViewHolderAdapter adapter=new ViewHolderAdapter(this);

    gridview.setAdapter(adapter);
    String urlImage="";//请自己选择网络上的静态图片

    for(int i=0;i<100;i++)
    {
    adapter.addItem(urlImage, "第"+i+"项");
    }

    }
    }

    原文:http://blog.csdn.net/hellogv/article/details/6541286#

  • 相关阅读:
    齐文词根词缀---3.12、ced-走
    感悟总结---2104012起(文字收集)
    齐文词根词缀---3.11、log说话
    齐文词根词缀---3.10、cata-向下(和cat,cad,cid,cis是一样的)
    齐文词根词缀---3.9、carn-肉、肉欲
    齐文词根词缀---3.8、capit-头(cap帽子)
    齐文词根词缀---3.7、-tect盖子
    齐文词根词缀---3.6、cant-唱、说
    齐文词根词缀---3.5、cad- / cas- 落下(【音变t-d-s】)
    CURL 发送POST请求
  • 原文地址:https://www.cnblogs.com/shanzei/p/2434510.html
Copyright © 2011-2022 走看看