Fragment中存放GridView,结合LruCache、DiskLruCache和HttpURLConnection实现图片下载、缓存

来源:互联网 发布:阿里云邀请码怎么获取 编辑:程序博客网 时间:2024/06/06 00:29

其实,在真正的项目实战当中如果仅仅是使用硬盘缓存的话,程序是有明显短板的。而如果只使用内存缓存的话,程序当然也会有很大的缺陷。因此,一个优秀的程序必然会将内存缓存和硬盘缓存结合到一起使用,那么本篇文章我们就来看一看,如何才能将LruCache和DiskLruCache完美结合到一起。

首先准备DiskLruCache.java文件。

Images.java:

[java] view plain copy
  1. public final static String[] imageThumbUrls = new String[] {  
  2.             "http://img.my.csdn.net/uploads/201308/31/1377949599_5269.jpg",  
  3.             "http://img.my.csdn.net/uploads/201308/31/1377949598_7858.jpg",  
  4.             "http://img.my.csdn.net/uploads/201308/31/1377949598_9982.jpg",  
  5.             "http://img.my.csdn.net/uploads/201308/31/1377949578_2770.jpg",  
  6.             "http://img.my.csdn.net/uploads/201308/31/1377949578_8744.jpg",  
  7.             "http://img.my.csdn.net/uploads/201308/31/1377949577_5210.jpg",  
  8.             "http://img.my.csdn.net/uploads/201308/31/1377949577_1998.jpg",  
  9.             "http://img.my.csdn.net/uploads/201308/31/1377949482_8813.jpg",  
  10.             "http://img.my.csdn.net/uploads/201308/31/1377949481_6577.jpg",  
  11.             "http://img.my.csdn.net/uploads/201308/31/1377949480_4490.jpg",  
  12.             "http://img.my.csdn.net/uploads/201308/31/1377949455_6792.jpg",  
  13.             "http://img.my.csdn.net/uploads/201308/31/1377949455_6345.jpg",  
  14.             "http://img.my.csdn.net/uploads/201308/31/1377949442_4553.jpg",  
  15.             "http://img.my.csdn.net/uploads/201308/31/1377949441_8987.jpg",  
  16.             "http://img.my.csdn.net/uploads/201308/31/1377949441_5454.jpg",  
  17.             "http://img.my.csdn.net/uploads/201308/31/1377949454_6367.jpg",  
  18.             "http://img.my.csdn.net/uploads/201308/31/1377949442_4562.jpg"  
  19.     };  

然后是fragment_subject.xml:

[html] view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2.   
  3. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  4.         android:orientation="vertical" android:layout_width="fill_parent"  
  5.         android:layout_height="wrap_content">  
  6.         <GridView  
  7.             android:id="@+id/photo_wall"  
  8.             android:layout_width="match_parent"  
  9.             android:layout_height="match_parent"  
  10.             android:columnWidth="@dimen/image_thumbnail_size"  
  11.             android:gravity="center"  
  12.             android:horizontalSpacing="@dimen/image_thumbnail_spacing"  
  13.             android:numColumns="auto_fit"  
  14.             android:stretchMode="columnWidth"  
  15.             android:verticalSpacing="@dimen/image_thumbnail_spacing" >  
  16.         </GridView>  
  17. </LinearLayout>  

接着我们要定义GridView中每一个子View的布局,新建一个fragment_subject_simple_item.xml布局(FrameLayout,可以在图片上显示文字),加入如下代码:

[html] view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>  
  2.   
  3. <FrameLayout  
  4.     xmlns:android="http://schemas.android.com/apk/res/android"  
  5.     android:layout_width="match_parent"  
  6.     android:layout_height="wrap_content">  
  7.     <ImageView  
  8.         android:id="@+id/photo"  
  9.         android:layout_width="match_parent"  
  10.         android:layout_height="match_parent"  
  11.         android:layout_centerInParent="true"  
  12.         android:scaleType="fitXY"  
  13.         />  
  14.     <TextView  
  15.         android:text="Heloo world"  
  16.         android:layout_width="fill_parent"  
  17.         android:layout_height="wrap_content"  
  18.         android:gravity="center"  
  19.         />  
  20.   
  21. </FrameLayout>  

新建PhotoWallAdapter做为GridView的适配器,代码如下所示:

[java] view plain copy
  1. package com.francis.changtravels.utils;  
  2.   
  3. import android.content.Context;  
  4. import android.content.pm.PackageInfo;  
  5. import android.content.pm.PackageManager;  
  6. import android.graphics.Bitmap;  
  7. import android.graphics.BitmapFactory;  
  8. import android.os.AsyncTask;  
  9. import android.os.Environment;  
  10. import android.util.LruCache;  
  11. import android.view.LayoutInflater;  
  12. import android.view.View;  
  13. import android.view.ViewGroup;  
  14. import android.widget.ArrayAdapter;  
  15. import android.widget.GridView;  
  16. import android.widget.ImageView;  
  17.   
  18. import java.io.BufferedInputStream;  
  19. import java.io.BufferedOutputStream;  
  20. import java.io.File;  
  21. import java.io.FileDescriptor;  
  22. import java.io.FileInputStream;  
  23. import java.io.IOException;  
  24. import java.io.OutputStream;  
  25. import java.net.HttpURLConnection;  
  26. import java.net.URL;  
  27. import java.security.MessageDigest;  
  28. import java.security.NoSuchAlgorithmException;  
  29. import java.util.HashSet;  
  30. import java.util.Set;  
  31. import com.francis.changtravels.utils.DiskLruCache;  
  32. import com.francis.changtravels.utils.DiskLruCache.Snapshot;  
  33.   
  34.   
  35.   
  36. import com.francis.changtravels.R;  
  37.   
  38.   
  39. /** 
  40.  * GridView的适配器,负责异步从网络上下载图片展示在照片墙上。 
  41.  * 
  42.  * @author guolin 
  43.  */  
  44. public class PhotoWallAdapter extends ArrayAdapter<String> {  
  45.   
  46.     /** 
  47.      * 记录所有正在下载或等待下载的任务。 
  48.      */  
  49.     private Set<BitmapWorkerTask> taskCollection;  
  50.   
  51.     /** 
  52.      * 图片缓存技术的核心类,用于缓存所有下载好的图片,在程序内存达到设定值时会将最少最近使用的图片移除掉。 
  53.      */  
  54.     private LruCache<String, Bitmap> mMemoryCache;  
  55.   
  56.     /** 
  57.      * 图片硬盘缓存核心类。 
  58.      */  
  59.     private DiskLruCache mDiskLruCache;  
  60.   
  61.     /** 
  62.      * GridView的实例 
  63.      */  
  64.     private GridView mPhotoWall;  
  65.   
  66.     /** 
  67.      * 记录每个子项的高度。 
  68.      */  
  69.     private int mItemHeight = 0;  
  70.   
  71.     public PhotoWallAdapter(Context context, int textViewResourceId, String[] objects,  
  72.                             GridView photoWall) {  
  73.         super(context, textViewResourceId, objects);  
  74.         mPhotoWall = photoWall;  
  75.         taskCollection = new HashSet<BitmapWorkerTask>();  
  76.         // 获取应用程序最大可用内存  
  77.         int maxMemory = (int) Runtime.getRuntime().maxMemory();  
  78.         int cacheSize = maxMemory / 8;  
  79.         // 设置图片缓存大小为程序最大可用内存的1/8  
  80.         mMemoryCache = new LruCache<String, Bitmap>(cacheSize) {  
  81.             @Override  
  82.             protected int sizeOf(String key, Bitmap bitmap) {  
  83.                 return bitmap.getByteCount();  
  84.             }  
  85.         };  
  86.         try {  
  87.             // 获取图片缓存路径  
  88.             File cacheDir = getDiskCacheDir(context, "thumb");  
  89.             if (!cacheDir.exists()) {  
  90.                 cacheDir.mkdirs();  
  91.             }  
  92.             // 创建DiskLruCache实例,初始化缓存数据  
  93.             mDiskLruCache = DiskLruCache  
  94.                     .open(cacheDir, getAppVersion(context), 110 * 1024 * 1024);  
  95.         } catch (IOException e) {  
  96.             e.printStackTrace();  
  97.         }  
  98.     }  
  99.   
  100.     @Override  
  101.     public View getView(int position, View convertView, ViewGroup parent) {  
  102.         final String url = getItem(position);  
  103.         View view;  
  104.         if (convertView == null) {  
  105.             view = LayoutInflater.from(getContext()).inflate(R.layout.fragment_subject_simple_item, null);  
  106.         } else {  
  107.             view = convertView;  
  108.         }  
  109.         final ImageView imageView = (ImageView) view.findViewById(R.id.photo);  
  110.         if (imageView.getLayoutParams().height != mItemHeight) {  
  111.             imageView.getLayoutParams().height = mItemHeight;  
  112.         }  
  113.         // 给ImageView设置一个Tag,保证异步加载图片时不会乱序  
  114.         imageView.setTag(url);  
  115.         imageView.setImageResource(R.drawable.empty_photo);  
  116.         loadBitmaps(imageView, url);  
  117.         return view;  
  118.     }  
  119.   
  120.     /** 
  121.      * 将一张图片存储到LruCache中。 
  122.      * 
  123.      * @param key 
  124.      *            LruCache的键,这里传入图片的URL地址。 
  125.      * @param bitmap 
  126.      *            LruCache的键,这里传入从网络上下载的Bitmap对象。 
  127.      */  
  128.     public void addBitmapToMemoryCache(String key, Bitmap bitmap) {  
  129.         if (getBitmapFromMemoryCache(key) == null) {  
  130.             mMemoryCache.put(key, bitmap);  
  131.         }  
  132.     }  
  133.   
  134.     /** 
  135.      * 从LruCache中获取一张图片,如果不存在就返回null。 
  136.      * 
  137.      * @param key 
  138.      *            LruCache的键,这里传入图片的URL地址。 
  139.      * @return 对应传入键的Bitmap对象,或者null。 
  140.      */  
  141.     public Bitmap getBitmapFromMemoryCache(String key) {  
  142.         return mMemoryCache.get(key);  
  143.     }  
  144.   
  145.     /** 
  146.      * 加载Bitmap对象。此方法会在LruCache中检查所有屏幕中可见的ImageView的Bitmap对象, 
  147.      * 如果发现任何一个ImageView的Bitmap对象不在缓存中,就会开启异步线程去下载图片。 
  148.      */  
  149.     public void loadBitmaps(ImageView imageView, String imageUrl) {  
  150.         try {  
  151.             Bitmap bitmap = getBitmapFromMemoryCache(imageUrl);  
  152.             if (bitmap == null) {  
  153.                 BitmapWorkerTask task = new BitmapWorkerTask();  
  154.                 taskCollection.add(task);  
  155.                 task.execute(imageUrl);  
  156.             } else {  
  157.                 if (imageView != null && bitmap != null) {  
  158.                     imageView.setImageBitmap(bitmap);  
  159.                 }  
  160.             }  
  161.         } catch (Exception e) {  
  162.             e.printStackTrace();  
  163.         }  
  164.     }  
  165.   
  166.     /** 
  167.      * 取消所有正在下载或等待下载的任务。 
  168.      */  
  169.     public void cancelAllTasks() {  
  170.         if (taskCollection != null) {  
  171.             for (BitmapWorkerTask task : taskCollection) {  
  172.                 task.cancel(false);  
  173.             }  
  174.         }  
  175.     }  
  176.   
  177.     /** 
  178.      * 根据传入的uniqueName获取硬盘缓存的路径地址。 
  179.      */  
  180.     public File getDiskCacheDir(Context context, String uniqueName) {  
  181.         String cachePath;  
  182.         if (Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())  
  183.                 || !Environment.isExternalStorageRemovable()) {  
  184.             cachePath = context.getExternalCacheDir().getPath();  
  185.         } else {  
  186.             cachePath = context.getCacheDir().getPath();  
  187.         }  
  188.         return new File(cachePath + File.separator + uniqueName);  
  189.     }  
  190.   
  191.     /** 
  192.      * 获取当前应用程序的版本号。 
  193.      */  
  194.     public int getAppVersion(Context context) {  
  195.         try {  
  196.             PackageInfo info = context.getPackageManager().getPackageInfo(context.getPackageName(),  
  197.                     0);  
  198.             return info.versionCode;  
  199.         } catch (PackageManager.NameNotFoundException e) {  
  200.             e.printStackTrace();  
  201.         }  
  202.         return 1;  
  203.     }  
  204.   
  205.     /** 
  206.      * 设置item子项的高度。 
  207.      */  
  208.     public void setItemHeight(int height) {  
  209.         if (height == mItemHeight) {  
  210.             return;  
  211.         }  
  212.         mItemHeight = height;  
  213.         notifyDataSetChanged();  
  214.     }  
  215.   
  216.     /** 
  217.      * 使用MD5算法对传入的key进行加密并返回。 
  218.      */  
  219.     public String hashKeyForDisk(String key) {  
  220.         String cacheKey;  
  221.         try {  
  222.             final MessageDigest mDigest = MessageDigest.getInstance("MD5");  
  223.             mDigest.update(key.getBytes());  
  224.             cacheKey = bytesToHexString(mDigest.digest());  
  225.         } catch (NoSuchAlgorithmException e) {  
  226.             cacheKey = String.valueOf(key.hashCode());  
  227.         }  
  228.         return cacheKey;  
  229.     }  
  230.   
  231.     /** 
  232.      * 将缓存记录同步到journal文件中。 
  233.      */  
  234.     public void fluchCache() {  
  235.         if (mDiskLruCache != null) {  
  236.             try {  
  237.                 mDiskLruCache.flush();  
  238.             } catch (IOException e) {  
  239.                 e.printStackTrace();  
  240.             }  
  241.         }  
  242.     }  
  243.   
  244.     private String bytesToHexString(byte[] bytes) {  
  245.         StringBuilder sb = new StringBuilder();  
  246.         for (int i = 0; i < bytes.length; i++) {  
  247.             String hex = Integer.toHexString(0xFF & bytes[i]);  
  248.             if (hex.length() == 1) {  
  249.                 sb.append('0');  
  250.             }  
  251.             sb.append(hex);  
  252.         }  
  253.         return sb.toString();  
  254.     }  
  255.   
  256.     /** 
  257.      * 异步下载图片的任务。 
  258.      * 
  259.      * @author guolin 
  260.      */  
  261.     class BitmapWorkerTask extends AsyncTask<String, Void, Bitmap> {  
  262.   
  263.         /** 
  264.          * 图片的URL地址 
  265.          */  
  266.         private String imageUrl;  
  267.   
  268.         @Override  
  269.         protected Bitmap doInBackground(String... params) {  
  270.             imageUrl = params[0];  
  271.             FileDescriptor fileDescriptor = null;  
  272.             FileInputStream fileInputStream = null;  
  273.             Snapshot snapShot = null;  
  274.             try {  
  275.                 // 生成图片URL对应的key  
  276.                 final String key = hashKeyForDisk(imageUrl);  
  277.                 // 查找key对应的缓存  
  278.                 snapShot = mDiskLruCache.get(key);  
  279.                 // 不用key 直接用imageUrl,imageUrl与缓存相对应  
  280.                 //snapShot = mDiskLruCache.get(imageUrl);  
  281.                 if (snapShot == null) {  
  282.                     // 如果没有找到对应的缓存,则准备从网络上请求数据,并写入缓存  
  283.                     DiskLruCache.Editor editor = mDiskLruCache.edit(key);  
  284.                    // DiskLruCache.Editor editor = mDiskLruCache.edit(imageUrl);  
  285.   
  286.                     if (editor != null) {  
  287.                         OutputStream outputStream = editor.newOutputStream(0);  
  288.                         if (downloadUrlToStream(imageUrl, outputStream)) {  
  289.                             editor.commit();  
  290.                         } else {  
  291.                             editor.abort();  
  292.                         }  
  293.                     }  
  294.                     // 缓存被写入后,再次查找key对应的缓存  
  295.                     snapShot = mDiskLruCache.get(key);  
  296.                     //snapShot = mDiskLruCache.get(imageUrl);  
  297.                 }  
  298.                 if (snapShot != null) {  
  299.                     fileInputStream = (FileInputStream) snapShot.getInputStream(0);  
  300.                     fileDescriptor = fileInputStream.getFD();  
  301.                 }  
  302.                 // 将缓存数据解析成Bitmap对象  
  303.                 Bitmap bitmap = null;  
  304.                 if (fileDescriptor != null) {  
  305.                     bitmap = BitmapFactory.decodeFileDescriptor(fileDescriptor);  
  306.                 }  
  307.                 if (bitmap != null) {  
  308.                     // 将Bitmap对象添加到内存缓存当中  
  309.                     addBitmapToMemoryCache(params[0], bitmap);  
  310.                 }  
  311.                 return bitmap;  
  312.             } catch (IOException e) {  
  313.                 e.printStackTrace();  
  314.             } finally {  
  315.                 if (fileDescriptor == null && fileInputStream != null) {  
  316.                     try {  
  317.                         fileInputStream.close();  
  318.                     } catch (IOException e) {  
  319.                     }  
  320.                 }  
  321.             }  
  322.             return null;  
  323.         }  
  324.   
  325.         @Override  
  326.         protected void onPostExecute(Bitmap bitmap) {  
  327.             super.onPostExecute(bitmap);  
  328.             // 根据Tag找到相应的ImageView控件,将下载好的图片显示出来。  
  329.             ImageView imageView = (ImageView) mPhotoWall.findViewWithTag(imageUrl);  
  330.             if (imageView != null && bitmap != null) {  
  331.                 imageView.setImageBitmap(bitmap);  
  332.             }  
  333.             taskCollection.remove(this);  
  334.         }  
  335.   
  336.         /** 
  337.          * 建立HTTP请求,并获取Bitmap对象。 
  338.          * 
  339.          * @param urlString 
  340.          *            图片的URL地址 
  341.          * @return 解析后的Bitmap对象 
  342.          */  
  343.         private boolean downloadUrlToStream(String urlString, OutputStream outputStream) {  
  344.             HttpURLConnection urlConnection = null;  
  345.             BufferedOutputStream out = null;  
  346.             BufferedInputStream in = null;  
  347.             try {  
  348.                 final URL url = new URL(urlString);  
  349.                 urlConnection = (HttpURLConnection) url.openConnection();  
  350.                 in = new BufferedInputStream(urlConnection.getInputStream(), 8 * 1024);  
  351.                 out = new BufferedOutputStream(outputStream, 8 * 1024);  
  352.                 int b;  
  353.                 while ((b = in.read()) != -1) {  
  354.                     out.write(b);  
  355.                 }  
  356.                 return true;  
  357.             } catch (final IOException e) {  
  358.                 e.printStackTrace();  
  359.             } finally {  
  360.                 if (urlConnection != null) {  
  361.                     urlConnection.disconnect();  
  362.                 }  
  363.                 try {  
  364.                     if (out != null) {  
  365.                         out.close();  
  366.                     }  
  367.                     if (in != null) {  
  368.                         in.close();  
  369.                     }  
  370.                 } catch (final IOException e) {  
  371.                     e.printStackTrace();  
  372.                 }  
  373.             }  
  374.             return false;  
  375.         }  
  376.   
  377.     }  
  378.   
  379. }  

代码有点长,我们一点点进行分析。首先在PhotoWallAdapter的构造函数中,我们初始化了LruCache类,并设置了内存缓存容量为程序最大可用内存的1/8,紧接着调用了DiskLruCache的open()方法来创建实例,并设置了硬盘缓存容量为10M,这样我们就把LruCache和DiskLruCache的初始化工作完成了。

接着在getView()方法中,我们为每个ImageView设置了一个唯一的Tag,这个Tag的作用是为了后面能够准确地找回这个ImageView,不然异步加载图片会出现乱序的情况。然后在getView()方法的最后调用了loadBitmaps()方法,加载图片的具体逻辑也就是在这里执行的了。

进入到loadBitmaps()方法中可以看到,实现是调用了getBitmapFromMemoryCache()方法来从内存中获取缓存,如果获取到了则直接调用ImageView的setImageBitmap()方法将图片显示到界面上。如果内存中没有获取到,则开启一个BitmapWorkerTask任务来去异步加载图片。

那么在BitmapWorkerTask的doInBackground()方法中,我们就灵活运用了上篇文章中学习的DiskLruCache的各种用法。首先根据图片的URL生成对应的MD5 key,然后调用DiskLruCache的get()方法来获取硬盘缓存,如果没有获取到的话则从网络上请求图片并写入硬盘缓存,接着将Bitmap对象解析出来并添加到内存缓存当中,最后将这个Bitmap对象显示到界面上,这样一个完整的流程就执行完了。

那么我们再来分析一下上述流程,每次加载图片的时候都优先去内存缓存当中读取,当读取不到的时候则回去硬盘缓存中读取,而如果硬盘缓存仍然读取不到的话,就从网络上请求原始数据。不管是从硬盘缓存还是从网络获取,读取到了数据之后都应该添加到内存缓存当中,这样的话我们下次再去读取图片的时候就能迅速从内存当中读取到,而如果该图片从内存中被移除了的话,那就重复再执行一遍上述流程就可以了。

这样我们就把LruCache和DiskLruCache完美结合到一起了。接下来还需要编写SubjectFragment的代码,非常简单,如下所示:

[java] view plain copy
  1. package com.francis.changtravels.fragment;  
  2.   
  3. import android.os.Bundle;  
  4. import android.support.v4.app.Fragment;  
  5. import android.util.DisplayMetrics;  
  6. import android.util.TypedValue;  
  7. import android.view.Gravity;  
  8. import android.view.LayoutInflater;  
  9. import android.view.View;  
  10. import android.view.ViewGroup;  
  11. import android.view.ViewTreeObserver;  
  12. import android.widget.ArrayAdapter;  
  13. import android.widget.GridView;  
  14. import android.widget.ListView;  
  15. import android.widget.SimpleAdapter;  
  16.   
  17.   
  18. import com.francis.changtravels.R;  
  19. import com.francis.changtravels.utils.Images;  
  20. import com.francis.changtravels.utils.PhotoWallAdapter;  
  21. import com.francis.changtravels.view.RefreshableView;  
  22.   
  23. import java.util.ArrayList;  
  24. import java.util.HashMap;  
  25. import java.util.List;  
  26. import java.util.Map;  
  27.   
  28.   
  29. /** 
  30.  * Created by Francis on 14-9-18. 
  31.  */  
  32. public class SubjectFragment extends Fragment {  
  33.   //  RefreshableView refreshableView;  
  34.   
  35.   
  36.     /** 
  37.      * 用于展示照片墙的GridView 
  38.      */  
  39.     private GridView mPhotoWall;  
  40.   
  41.     /** 
  42.      * GridView的适配器 
  43.      */  
  44.     private PhotoWallAdapter mAdapter;  
  45.   
  46.     private int mImageThumbSize;  
  47.     private int mImageThumbSpacing;  
  48.   
  49.     @Override  
  50.     public View onCreateView(LayoutInflater inflater, ViewGroup container,  
  51.                              Bundle savedInstanceState) {  
  52.   
  53.   
  54.         View rootView = inflater.inflate(R.layout.fragment_subject,container,false);  
  55.   
  56.         mImageThumbSize = getResources().getDimensionPixelSize(  
  57.                 R.dimen.image_thumbnail_size);  
  58.         mImageThumbSpacing = getResources().getDimensionPixelSize(  
  59.                 R.dimen.image_thumbnail_spacing);  
  60.   
  61.         mPhotoWall = (GridView) rootView.findViewById(R.id.photo_wall);  
  62.         mAdapter = new PhotoWallAdapter(getActivity(), 0, Images.imageThumbUrls,  
  63.                 mPhotoWall);  
  64.         mPhotoWall.setAdapter(mAdapter);  
  65.         mPhotoWall.getViewTreeObserver().addOnGlobalLayoutListener(  
  66.                 new ViewTreeObserver.OnGlobalLayoutListener() {  
  67.   
  68.                     @Override  
  69.                     public void onGlobalLayout() {  
  70.                         final int numColumns = (int) Math.floor(mPhotoWall  
  71.                                 .getWidth()  
  72.                                 / (mImageThumbSize + mImageThumbSpacing));  
  73.                         if (numColumns > 0) {  
  74.                             int columnWidth = (mPhotoWall.getWidth() / numColumns)  
  75.                                     - mImageThumbSpacing;  
  76.                             mAdapter.setItemHeight(columnWidth);  
  77.                             mPhotoWall.getViewTreeObserver()  
  78.                                     .removeGlobalOnLayoutListener(this);  
  79.                         }  
  80.                     }  
  81.                 });  
  82.   
  83.         return rootView;  
  84.     }  
  85.   
  86.     @Override  
  87.     public void onPause() {  
  88.         super.onPause();  
  89.         mAdapter.fluchCache();  
  90.     }  
  91.   
  92.     @Override  
  93.     public void onDestroyView() {  
  94.         super.onDestroyView();  
  95.         // 退出程序时结束所有的下载任务  
  96.         mAdapter.cancelAllTasks();  
  97.     }  
  98. }  

上述代码中,我们通过getViewTreeObserver()的方式监听View的布局事件,当布局完成以后,我们重新修改一下GridView中子View的高度,以保证子View的宽度和高度可以保持一致。

到这里还没有结束,最后还需要配置一下AndroidManifest.xml文件,并加入相应的权限,如下所示:

[html] view plain copy
  1. <uses-permission android:name="android.permission.INTERNET" />    
  2. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />   
0 0
原创粉丝点击