图片的内存缓存控制

news/2024/7/8 6:56:50

为什么80%的码农都做不了架构师?>>>   hot3.png

从网上找到的,自己整理了下:


public class MemoryCache
    {
        private static final String TAG = "MemoryCache";
        /**
         * 放入缓存时是个同步操作
         * LinkedHashMap构造方法的最后一个参数true代表这个map里的元素将按照最近使用次数由少到多排列,即LRU
         * 这样的好处是如果要将缓存中的元素替换,则先遍历出最近最少使用的元素来替换以提高效率
         */
        private Map<String, Bitmap> mCache = Collections.synchronizedMap(new LinkedHashMap<String, Bitmap>(10, 1.5f, true));
        /**
         * 缓存中图片所占用的字节,初始0,将通过此变量严格控制缓存所占用的堆内存
         */
        private long mSize = 0;
        /**
         * 缓存只能占用的最大堆内存
         */
        private long mLimit = 1000000;

        public MemoryCache() {
            setLimit(Runtime.getRuntime().maxMemory() / 10);
        }

        /**
         * 设置内存使用上限
         * @param limit
         */
        public void setLimit(long limit)  {
            mLimit = limit;
            Log.i(TAG, "MemoryCache will use up to " + (mLimit / 1024. / 1024.) + "MB");
        }

        /**
         * 获取内存中缓存的图片资源
         *
         * @param key
         * @return
         */
        public Bitmap get(String key) {
            try {
                if (! mCache.containsKey(key)) return null;
                return mCache.get(key);
            } catch (NullPointerException ex) {
                return null;
            }
        }

        /**
         * 将图片资源缓存到内存
        *
         * @param key
         * @param bitmap
         */
        public void put(String key, Bitmap bitmap) {
            try {
            if (mCache.containsKey(key))
                mSize -= getSizeInBytes(mCache.get(key));
                mCache.put(key, bitmap);
                // 累计当前缓存已使用的内存大小
                mSize += getSizeInBytes(bitmap);
                checkSize();
            } catch (Throwable th) {
                th.printStackTrace();
            }
        }

        /**
         * 检测图片缓存的内存占用
         */
        private void checkSize()
        {
            Log.i(TAG, "cache size=" + mSize + " length=" + mCache.size());
            if (mSize > mLimit) {
                // 先遍历最近最少使用的元素
                Iterator<Map.Entry<String, Bitmap>> iterator = mCache.entrySet().iterator();
                while (iterator.hasNext()) {
                    Map.Entry<String, Bitmap> entry = iterator.next();
                    mSize -= getSizeInBytes(entry.getValue());
                    iterator.remove();
                    if (mSize <= mLimit)
                        break;
                }
                Log.i(TAG, "Clean cache. New size " + mCache.size());
            }
        }

        /**
         * 清空内存缓存
         */
        public void clear() {
            mCache.clear();
        }

        /**
         * 图片占用的内存
         *
         * @param bitmap
         * @return
         */
        long getSizeInBytes(Bitmap bitmap) {
            if (bitmap == null)
                return 0;
            return bitmap.getRowBytes() * bitmap.getHeight();
        }
    }

首先限制内存图片缓冲的堆内存大小,每次有图片往缓存里加时判断是否超过限制大小,超过的话就从中取出最少使用的图片并将其移除,当然这里如果不采用这种方式,换做软引用也是可行的,二者目的皆是最大程度的利用已存在于内存中的图片缓存,避免重复制造垃圾增加GC负担,OOM溢出往往皆因内存瞬时大量增加而垃圾回收不及时造成的。只不过二者区别在于LinkedHashMap里的图片缓存在没有移除出去之前是不会被GC回收的,而SoftReference里的图片缓存在没有其他引用保存时随时都会被GC回收。所以在使用LinkedHashMap这种LRU算法缓存更有利于图片的有效命中,当然二者配合使用的话效果更佳,即从LinkedHashMap里移除出的缓存放到SoftReference里,这就是内存的二级缓存,有兴趣的童鞋不凡一试。

原文:http://www.eoeandroid.com/thread-254866-1-1.html

转载于:https://my.oschina.net/zhouz/blog/213181


http://www.niftyadmin.cn/n/4072358.html

相关文章

DES算法实例详解

为什么80%的码农都做不了架构师&#xff1f;>>> 译自J. Orlin Grabbe的名作《DES Algorithm Illustrated》&#xff0c;国外许多大学将该文章作为补充材料&#xff0c;可作为理解DES算法的最佳入门手册。反观许多教材介绍DES时直接照搬一张流程图&#xff0c;图中I…

手把手教你如何正确启动Android SDK 1.5模拟器

为什么80%的码农都做不了架构师&#xff1f;>>> 首先要确认下自己的电脑是否安装过JDK、没有的话可以 点我下载&#xff08;右键迅雷下载&#xff09; &#xff01;安装好JDK后&#xff0c;按住键盘上的 WINR 键&#xff0c;在弹出的运行框中输入cmd&#xff0c;回…

《深入java虚拟机》读后(2)

三、OutOfMemoryError 第三章 垃圾收集器与内存分配策略 一、 判断对象是否被实用 1、 引用计数法 给对象添加一个引用计数器&#xff0c;每当有一个地方引用他&#xff0c;计数器就加一&#xff0c;引用失效就减一&#xff0c;计数器为零的对象就是不再被实用的。&#xff08;…

Tomcat7.0配置

2019独角兽企业重金招聘Python工程师标准>>> Tomcat7.0下载地址&#xff1a;http://tomcat.apache.org/download-70.cgi 选择符合自己操作系统的版本即可(本机win8-64位系统)&#xff0c;选择如下&#xff1a; http://118.186.9.91/files/40150000012D8243/123.125.…

Tomcat 6.0 简介

2019独角兽企业重金招聘Python工程师标准>>> 本片翻译来自&#xff1a;http://tomcat.apache.org/tomcat-6.0-doc/introduction.html 介绍 无论是开发者还是tomcat管理员在使用前都需要了解一些必要的信息&#xff0c;本篇简单的介绍tomcat中的一些术语和概念。 比如…

浅谈自己对前端MVC的理解

一、MVC的定义 MVC全名Model View Controller&#xff0c;是模型(model)&#xff0d;视图(view)&#xff0d;控制器(controller)的缩写。MVC这一概念是来源于后端的框架构建思想&#xff0c;是一种软件设计典范&#xff0c;用一种业务逻辑、数据、界面显示分离的方法组织代码&a…

react: nextJs koa project basic structure

1、init nextJs project npm init npm install react react-dom next config script in package.json "dev": "next""start": "next start" "build": "next build" npm run dev result: 404 page not found 2、in…

【目录】java学习路径

讲解java的博客&#xff0c;从零到一的过程 。 01-JavaSE基础 02-HTML 03- css 04- JavaScript 05- MySQL 06- jdbc 07- xml 08- tomcat 09- servlet和jsp 等转载于:https://www.cnblogs.com/Chamberlain/p/11032965.html