13.缓存、三级缓存、内存溢出、AsyncTask

SharePreference工具类

  1. /**
  2. * SharePreference封装
  3. *
  4. */
  5. public class PrefUtils {
  6. public static final String PREF_NAME = "config";
  7. public static boolean getBoolean(Context ctx, String key,
  8. boolean defaultValue) {
  9. SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
  10. Context.MODE_PRIVATE);
  11. return sp.getBoolean(key, defaultValue);
  12. }
  13. public static void setBoolean(Context ctx, String key, boolean value) {
  14. SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
  15. Context.MODE_PRIVATE);
  16. sp.edit().putBoolean(key, value).commit();
  17. }
  18. public static String getString(Context ctx, String key, String defaultValue) {
  19. SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
  20. Context.MODE_PRIVATE);
  21. return sp.getString(key, defaultValue);
  22. }
  23. public static void setString(Context ctx, String key, String value) {
  24. SharedPreferences sp = ctx.getSharedPreferences(PREF_NAME,
  25. Context.MODE_PRIVATE);
  26. sp.edit().putString(key, value).commit();
  27. }
  28. }

缓存工具类

  1. public class CacheUtils {
  2. /**
  3. * 缓存原理:设置缓存 key 是url, value是json(解析出来的)
  4. */
  5. public static void setCache(String key, String value, Context ctx) {
  6. PrefUtils.setString(ctx, key, value);
  7. // 可以将缓存放在文件中, 文件名就是Md5(url), 文件内容是json
  8. }
  9. /**
  10. * 获取缓存 key 是url
  11. */
  12. public static String getCache(String key, Context ctx) {
  13. return PrefUtils.getString(ctx, key, null);
  14. }
  15. }

用法:

1.在请求完网络,获取json数据后保存起来

  1. private void getDataFromServer() {
  2. HttpUtils utils = new HttpUtils();
  3. utils.send(HttpMethod.GET, GlobalContants.PHOTOS_URL,
  4. new RequestCallBack<String>() {
  5. @Override
  6. public void onSuccess(ResponseInfo<String> responseInfo) {
  7. String result = (String) responseInfo.result;
  8. parseData(result);
  9. // 设置缓存
  10. CacheUtils.setCache(GlobalContants.PHOTOS_URL, result,
  11. mActivity);
  12. }
  13. @Override
  14. public void onFailure(HttpException error, String msg) {
  15. Toast.makeText(mActivity, msg, Toast.LENGTH_SHORT)
  16. .show();
  17. error.printStackTrace();
  18. }
  19. });
  20. }

2.在初始化数据的时候判断,可以直接解析数据,也可以什么都不做,然后获取网络数据看有没有最新的

  1. public void initData() {
  2. String cache = CacheUtils
  3. .getCache(GlobalContants.PHOTOS_URL, mActivity);
  4. if (!TextUtils.isEmpty(cache)) {
  5. // parseData(cache);
  6. }
  7. getDataFromServer();
  8. }

解析数据

  1. protected void parseData(String result) {
  2. Gson gson = new Gson();
  3. PhotosData data = gson.fromJson(result, PhotosData.class);
  4. mPhotoList = data.data.news;// 获取组图列表集合
  5. if (mPhotoList != null) {
  6. mAdapter = new PhotoAdapter();
  7. lvPhoto.setAdapter(mAdapter);
  8. gvPhoto.setAdapter(mAdapter);
  9. }
  10. }

三级缓存



- 内存缓存, 优先加载, 速度最快
- 本地缓存, 次优先加载, 速度快
- 网络缓存, 不优先加载, 速度慢,浪费流量

内存溢出OOM

- 内存缓存的时候可能内存溢出,因为Android默认给每个app只分配16M的内存

解决方法1:java中的引用(使用软引用)

- 强引用 垃圾回收器不会回收, java默认引用都是强引用

- 软引用 SoftReference   在内存不够时,垃圾回收器会考虑回收

- 弱引用 WeakReference  在内存不够时,垃圾回收器会优先回收

- 虚引用 PhantomReference  在内存不够时,垃圾回收器最优先回收

注意: Android2.3+, 系统会优先将SoftReference的对象提前回收掉, 即使内存够用

解决方法2:LruCache

least recentlly use 最少最近使用算法

会将内存控制在一定的大小内, 超出最大值时会自动回收, 这个最大值开发者自己定

解决方法3:图片压缩

xutils就是这个原理,现在不用xutils,现在自定义几个工具类

1.自定义加载工具类

  1. public class MyBitmapUtils {
  2. NetCacheUtils mNetCacheUtils;
  3. LocalCacheUtils mLocalCacheUtils;
  4. MemoryCacheUtils mMemoryCacheUtils;
  5. public MyBitmapUtils() {
  6. mMemoryCacheUtils = new MemoryCacheUtils();
  7. mLocalCacheUtils = new LocalCacheUtils();
  8. mNetCacheUtils = new NetCacheUtils(mLocalCacheUtils, mMemoryCacheUtils);
  9. }
  10. public void display(ImageView ivPic, String url) {
  11. ivPic.setImageResource(R.drawable.news_pic_default);// 设置默认加载图片
  12. Bitmap bitmap = null;
  13. // 从内存读
  14. bitmap = mMemoryCacheUtils.getBitmapFromMemory(url);
  15. if (bitmap != null) {
  16. ivPic.setImageBitmap(bitmap);
  17. System.out.println("从内存读取图片啦...");
  18. return;
  19. }
  20. // 从本地读
  21. bitmap = mLocalCacheUtils.getBitmapFromLocal(url);
  22. if (bitmap != null) {
  23. ivPic.setImageBitmap(bitmap);
  24. System.out.println("从本地读取图片啦...");
  25. mMemoryCacheUtils.setBitmapToMemory(url, bitmap);// 将图片保存在内存
  26. return;
  27. }
  28. // 从网络读
  29. mNetCacheUtils.getBitmapFromNet(ivPic, url);
  30. }
  31. }

2.网络缓存、AsyncTask

  1. public class NetCacheUtils {
  2. private LocalCacheUtils mLocalCacheUtils;
  3. private MemoryCacheUtils mMemoryCacheUtils;
  4. public NetCacheUtils(LocalCacheUtils localCacheUtils,
  5. MemoryCacheUtils memoryCacheUtils) {
  6. mLocalCacheUtils = localCacheUtils;
  7. mMemoryCacheUtils = memoryCacheUtils;
  8. }
  9. /**
  10. * 从网络下载图片
  11. *
  12. * @param ivPic
  13. * @param url
  14. */
  15. public void getBitmapFromNet(ImageView ivPic, String url) {
  16. new BitmapTask().execute(ivPic, url);// 启动AsyncTask,
  17. // 参数会在doInbackground中获取
  18. }
  19. /**
  20. * Handler和线程池的封装
  21. *
  22. * 第一个泛型: 参数类型 第二个泛型: 更新进度的泛型, 第三个泛型是onPostExecute的返回结果
  23. *
  24. * @author Kevin
  25. *
  26. */
  27. class BitmapTask extends AsyncTask<Object, Void, Bitmap> {
  28. private ImageView ivPic;
  29. private String url;
  30. /**
  31. * 后台耗时方法在此执行, 子线程
  32. */
  33. @Override
  34. protected Bitmap doInBackground(Object... params) {
  35. ivPic = (ImageView) params[0];
  36. url = (String) params[1];
  37. ivPic.setTag(url);// 将url和imageview绑定
  38. return downloadBitmap(url);
  39. }
  40. /**
  41. * 更新进度, 主线程
  42. */
  43. @Override
  44. protected void onProgressUpdate(Void... values) {
  45. super.onProgressUpdate(values);
  46. }
  47. /**
  48. * 耗时方法结束后,执行该方法, 主线程
  49. */
  50. @Override
  51. protected void onPostExecute(Bitmap result) {
  52. if (result != null) {
  53. String bindUrl = (String) ivPic.getTag();
  54. if (url.equals(bindUrl)) {// 确保图片设定给了正确的imageview
  55. ivPic.setImageBitmap(result);
  56. mLocalCacheUtils.setBitmapToLocal(url, result);// 将图片保存在本地
  57. mMemoryCacheUtils.setBitmapToMemory(url, result);// 将图片保存在内存
  58. System.out.println("从网络缓存读取图片啦...");
  59. }
  60. }
  61. }
  62. }
  63. /**
  64. * 下载图片
  65. *
  66. * @param url
  67. * @return
  68. */
  69. private Bitmap downloadBitmap(String url) {
  70. HttpURLConnection conn = null;
  71. try {
  72. conn = (HttpURLConnection) new URL(url).openConnection();
  73. conn.setConnectTimeout(5000);
  74. conn.setReadTimeout(5000);
  75. conn.setRequestMethod("GET");
  76. conn.connect();
  77. int responseCode = conn.getResponseCode();
  78. if (responseCode == 200) {
  79. InputStream inputStream = conn.getInputStream();
  80. //图片压缩处理
  81. BitmapFactory.Options option = new BitmapFactory.Options();
  82. option.inSampleSize = 2;//宽高都压缩为原来的二分之一, 此参数需要根据图片要展示的大小来确定
  83. option.inPreferredConfig = Bitmap.Config.RGB_565;//设置图片格式
  84. Bitmap bitmap = BitmapFactory.decodeStream(inputStream, null, option);
  85. return bitmap;
  86. }
  87. } catch (Exception e) {
  88. e.printStackTrace();
  89. } finally {
  90. conn.disconnect();
  91. }
  92. return null;
  93. }
  94. }

3.本地缓存(SD卡),以MD5加密为文件名,MD5在附件

  1. public class LocalCacheUtils {
  2. public static final String CACHE_PATH = Environment
  3. .getExternalStorageDirectory().getAbsolutePath() + "/zhbj_cache_52";
  4. /**
  5. * 从本地sdcard读图片
  6. */
  7. public Bitmap getBitmapFromLocal(String url) {
  8. try {
  9. String fileName = MD5Encoder.encode(url);
  10. File file = new File(CACHE_PATH, fileName);
  11. if (file.exists()) {
  12. Bitmap bitmap = BitmapFactory.decodeStream(new FileInputStream(
  13. file));//decodeStream放的是输入输出流
  14. return bitmap;
  15. }
  16. } catch (Exception e) {
  17. e.printStackTrace();
  18. }
  19. return null;
  20. }
  21. /**
  22. * 向sdcard写图片
  23. *
  24. * @param url
  25. * @param bitmap
  26. */
  27. public void setBitmapToLocal(String url, Bitmap bitmap) {
  28. try {
  29. String fileName = MD5Encoder.encode(url);
  30. File file = new File(CACHE_PATH, fileName);
  31. File parentFile = file.getParentFile();
  32. if (!parentFile.exists()) {// 如果文件夹不存在, 创建文件夹
  33. parentFile.mkdirs();
  34. }
  35. // 将图片保存在本地
  36. bitmap.compress(CompressFormat.JPEG, 100,
  37. new FileOutputStream(file));//100是质量
  38. } catch (Exception e) {
  39. e.printStackTrace();
  40. }
  41. }
  42. }

3.内存缓存:

  1. public class MemoryCacheUtils {
  2. // private HashMap<String, SoftReference<Bitmap>> mMemoryCache = new
  3. // HashMap<String, SoftReference<Bitmap>>();//一开始使用map,后来使用软引用
  4. private LruCache<String, Bitmap> mMemoryCache;
  5. public MemoryCacheUtils() {
  6. long maxMemory = Runtime.getRuntime().maxMemory() / 8;//主流都是分配16m的8/1
  7. mMemoryCache = new LruCache<String, Bitmap>((int) maxMemory) {
  8. @Override
  9. protected int sizeOf(String key, Bitmap value) {
  10. int byteCount = value.getRowBytes() * value.getHeight();// 获取图片占用内存大小
  11. return byteCount;
  12. }
  13. };
  14. }
  15. /**
  16. * 从内存读
  17. *
  18. * @param url
  19. */
  20. public Bitmap getBitmapFromMemory(String url) {
  21. // SoftReference<Bitmap> softReference = mMemoryCache.get(url);
  22. // if (softReference != null) {
  23. // Bitmap bitmap = softReference.get();
  24. // return bitmap;
  25. // }
  26. return mMemoryCache.get(url);
  27. }
  28. /**
  29. * 写内存
  30. *
  31. * @param url
  32. * @param bitmap
  33. */
  34. public void setBitmapToMemory(String url, Bitmap bitmap) {
  35. // SoftReference<Bitmap> softReference = new
  36. // SoftReference<Bitmap>(bitmap);
  37. // mMemoryCache.put(url, softReference);
  38. mMemoryCache.put(url, bitmap);
  39. }
  40. }

来自为知笔记(Wiz)

附件列表

时间: 2024-12-20 12:24:50

13.缓存、三级缓存、内存溢出、AsyncTask的相关文章

图片缓存----三级缓存

一.适配器类中图片设置 ImageCacheTask imageCache = new ImageCacheTas(context); final String imgCover = list.get(position).getCover();        vh.img_movie_item_listview.setTag(imgCover);        Bitmap bm = imageCache.loadImage(imgCover, new ImageCallche() {     

android加载大量图片内存溢出的三种方法

android加载大量图片内存溢出的三种解决办法 方法一:  在从网络或本地加载图片的时候,只加载缩略图. /** * 按照路径加载图片 * @param path 图片资源的存放路径 * @param scalSize 缩小的倍数 * @return */ public static Bitmap loadResBitmap(String path, int scalSize) { BitmapFactory.Options options = new BitmapFactory.Option

安卓网络请求图片到图片的三级缓存技术(内存缓存,本地缓存,网络缓存)

安卓网络请求图片,对于我们来说并不陌生,因为每个应用都有可能会用到这一技术.通常情况下,我们第一次都是从网络上请求图片资源,然后将 图片资源保存到内存和本地,下一次动态显示图片的时候就不需要再从网络上请求图片资源了,直接从本地或者内存中获取就可以了.这就涉及到图片 的三级缓存技术,分别是内存缓存,本地缓存,网络缓存. 缓存的流程图: 首先我们定义一个类叫ClassLoader: package com.jsako.showprodinfodemo; import java.io.FileOutp

利用LruCache进行图片的内存缓存,避免OOM异常(内存溢出异常)

本文将介绍图片的内存缓存技术:LruCache和软引用的使用 1.软引用 我们一般用内存来缓存图片都是通过使用HashMap这个集合来保存图片的Url和图片对应的Bitmap对象 ,但是当我们需要缓存很多图片的时候,那么问题就来了,由于Map里面保存的是Bitmap对象的强引用, 所有导致dvm虚拟机的垃圾回收机制在内存溢出的时候无法回收Map里面的对象,导致内存溢出,程序就会 崩掉.造成这样的就是因为Map里面保存的是Bitmap对象的强引用导致的. 如何解决这一问题?其实很简单,map里面的

android文件缓存,并SD卡创建目录未能解决和bitmap内存溢出解决

1.相关代码: 加入权限: <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> /** 获取SD卡路径 **/ private static String getSDPath() { St

Android开发中图片的三级缓存策略

一.简介 现在的Android应用程序中,不可避免的都会使用到图片,如果每次加载图片的时候都要从网络重新拉取,这样不但很耗费用户的流量,而且图片加载的也会很慢,用户体验很不好.所以一个应用的图片缓存策略是很重要的.通常情况下,Android应用程序中图片的缓存策略采用"内存-本地-网络"三级缓存策略,首先应用程序访问网络拉取图片,分别将加载的图片保存在本地SD卡中和内存中,当程序再一次需要加载图片的时候,先判断内存中是否有缓存,有则直接从内存中拉取,否则查看本地SD卡中是否有缓存,SD

Android中图片的三级缓存策略

一.简介 现在的Android应用程序中,不可避免的都会使用到图片,如果每次加载图片的时候都要从网络重新拉取,这样不但很耗费用户的流量,而且图片加载的也会很慢,用户体验很不好.所以一个应用的图片缓存策略是很重要的.通常情况下,Android应用程序中图片的缓存策略采用"内存-本地-网络"三级缓存策略,首先应用程序访问网络拉取图片,分别将加载的图片保存在本地SD卡中和内存中,当程序再一次需要加载图片的时候,先判断内存中是否有缓存,有则直接从内存中拉取,否则查看本地SD卡中是否有缓存,SD

关于Android中的三级缓存

三级缓存的提出就是为了提升用户体验.当我们第一次打开应用获取图片时,先到网络去下载图片,然后依次存入内存缓存,磁盘缓存,当我们再一次需要用到刚才下载的这张图片时,就不需要再重复的到网络上去下载,直接可以从内存缓存和磁盘缓存中找,由于内存缓存速度较快,我们优先到内存缓存中寻找该图片,如果找到则运用,如果没有找到(内存缓存大小有限),那么我们再到磁盘缓存中去找.只要我们合理的去协调这三层缓存运用,便可以提升应用性能.三级缓存指的是:内存缓存.本地缓存.网络缓存.其各自的特点是内存缓存速度快, 优先读

ImageLoad(三级缓存(NetCache,MemoryCache,DiskCache),开源框架,OOM)

ImageLoad 原创作品:未经本人允许,不得转载 前段时间写项目时遇到了一个问题,就是从网络获取图片资源的问题,总是出现OOM异常,经过几天的努力,终于处理的还算是可以使用,OOM的处理一直都是很头疼的问题.对于三级缓存的处理也是每个项目所必不可少的,所以我打算把我的学习心的写下来,以下主要针对三级缓存的原理,以及一些知识点进行详细的叙述,有不对的地方还希望大家能多加点评,指正 一.前言 首先再说三级缓存之前我们首先了解一下垃圾回收机制 下面是目前我所了解的几种引用关系 (一).强引用(St