先上效果图:

如图,android默认也有Gallery,很多软件在调用时,都是使用自己的Gallery,一方面好维护,另外一方面可以做优化。要做成以上样式,图片加载类起至关重要,一不小心,就好OOM, 下面这个类就是做Gallery的核心。

package com.example.gallery.utils;

import java.lang.reflect.Field;
import java.util.LinkedList;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Semaphore;

import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.BitmapFactory.Options;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.support.v4.util.LruCache;
import android.util.DisplayMetrics;
import android.view.ViewGroup.LayoutParams;
import android.widget.ImageView;

public class ImageLoader {

    public static ImageLoader mInstance;
    private LruCache<String, Bitmap> mLruCache;
    private ExecutorService mThreadPool;
    private static final int DEFAULT_THREAD_POOL_SIZE = 1;
    private Type mType = Type.LIFO; //队列的调度方式
    private LinkedList<Runnable> mTaskQueue;  //任务队列,可以从头部和尾部取对象,链表不用连续的内存
    private Thread mPoolThread; //后台轮询线程
    private Handler mPoolThreadHandler;
    private Handler mUIHandler; //UI线程
    private Semaphore mSemaphorePoolThreadHandler = new Semaphore(0); //信号量用来同步,默认申请0
    private Semaphore mSemaphoreThreadPool;

    public enum Type {
        FIFO,LIFO;
    }
    private ImageLoader(int threadCount, Type type) {
        init(threadCount, type);
    }

    private void init(int threadCount, Type type) {
        //后台轮询线程
        mPoolThread = new Thread() {
            @Override
            public void run() {
                Looper.prepare();
                mPoolThreadHandler = new Handler(){
                    public void handleMessage(android.os.Message msg) {
                        //线程池去取出一个任务进行执行
                        mThreadPool.execute(getTask());
                        try {
                            mSemaphoreThreadPool.acquire(); //阻塞住
                        } catch (InterruptedException e) {
                            e.printStackTrace();
                        }
                    };
                };
                mSemaphorePoolThreadHandler.release();//释放信号量
                Looper.loop();
            };
        };
        mPoolThread.start();
        //获取我们应用的最大可用内存
        int maxMemory = (int) Runtime.getRuntime().maxMemory();
        int cacheMemory = maxMemory / 8;
        mLruCache = new LruCache<String, Bitmap>(cacheMemory){
            protected int sizeOf(String key, Bitmap value) {

                return value.getRowBytes() * value.getHeight();
            };// 每行的字节数*高度
        };
        //创建线程池
        mThreadPool = Executors.newFixedThreadPool(threadCount);
        mTaskQueue = new LinkedList<Runnable>();
        mType = type;
        mSemaphoreThreadPool = new Semaphore(threadCount);
    }

    public static ImageLoader getInstance(int size, Type type) {
        if(mInstance == null) {
            //效率的提升,如果多个线程进入时
            synchronized (ImageLoader.class) {
                if(mInstance == null) { // 每次都new ,会产生多个对象
                    mInstance = new ImageLoader(DEFAULT_THREAD_POOL_SIZE, Type.LIFO);
                }
            }
        }
        return mInstance;
    }

    public void loadImage(final String path, final ImageView image) {
        image.setTag(path);//防止多次复用
        if(mUIHandler == null) {
            mUIHandler = new Handler() {
                public void handleMessage(android.os.Message msg) {
                    //获取得到图片,为image回调设置图片
                    ImageBeanHolder holder = (ImageBeanHolder) msg.obj;
                    Bitmap bm = holder.bitmap;
                    ImageView imageview = holder.image;
                    String path = holder.path;
                    //将path与getTag存储路径进行比较
                    if(imageview.getTag().toString().equals(path)) {
                        imageview.setImageBitmap(bm);
                    }

                };
            };
        }
        Bitmap bm = getBitmapFromLruCache(path);
        if (bm != null) {
            refreshBitmap(path, image, bm);
        } else {
            addTask(new Runnable() {

                @Override
                public void run() {
                    //加载图片
                    //图片的压缩
                    //1.获得图片需要显示的大小
                    ImageSize imageViewSize = getImageViewSize(image);
                    //2.压缩图片
                    Bitmap bm = decodeSampleBitmapFromPath(path, imageViewSize.width, imageViewSize.height);
                    //3.把图片加入到缓存
                    addBitmapToLruCache(path,bm);
                    //4.进行回调
                    refreshBitmap(path, image, bm);
                    mSemaphoreThreadPool.release(); //任务完成,就施放信号量
                }
            });
        }
    }

    //从任务队列中取任务
    private Runnable getTask() {
        if(mType == Type.FIFO) {
            return mTaskQueue.removeFirst();
        } else if(mType == Type.LIFO) {
            return mTaskQueue.removeLast();
        }
        return null;
    }

    private void refreshBitmap(final String path, final ImageView image, Bitmap bm) {
        Message msg = Message.obtain();
        ImageBeanHolder holder = new ImageBeanHolder();
        holder.bitmap = bm;
        holder.image = image;
        holder.path = path;
        msg.obj = holder;
        mUIHandler.sendMessage(msg);
    }

    //将图片加到LruCache
    private void addBitmapToLruCache(String path, Bitmap bm) {
        if(getBitmapFromLruCache(path) == null) {
            if(bm != null) {
                mLruCache.put(path, bm);
            }
        }
    }

    private ImageSize getImageViewSize(ImageView image) {
        ImageSize imageSize = new ImageSize();
        DisplayMetrics displayMetrics = image.getContext().getResources().getDisplayMetrics();
        LayoutParams layoutParams = image.getLayoutParams();
        int width = image.getWidth(); //获取实际宽度
//        int width = getImageViewFiledValue(image, "mMaxWidth");
        //layoutParams.width == LayoutParams.WRAP_CONTENT ? 0 :
        if(width <= 0) { //wrap_content -1 fill_parent -2
            width = layoutParams.width; //获取image在layout中声明的宽度
        }
        if(width <= 0) {
            width = getImageViewFiledValue(image, "mMaxWidth"); //检查最大值
        }
        if(width <= 0) {
            width = displayMetrics.widthPixels;//为屏幕的宽度
        }

        int height = image.getHeight(); //获取实际高度
        //layoutParams.width == LayoutParams.WRAP_CONTENT ? 0 :
        if(height <= 0) { //wrap_content -1 fill_parent -2
            height = layoutParams.height; //获取image在layout中声明的高度
        }
        if(height <= 0) {
            height = getImageViewFiledValue(image, "mMaxHeight"); //检查最大值
        }
        if(height <= 0) {
            height = displayMetrics.heightPixels;//为屏幕的高度
        }
        imageSize.height = height;
        imageSize.width = width;
        return imageSize;
    }

    //为什么用反射,因为检查最大值是API 16才能用,兼容API 8 时,就用反射
    private static int getImageViewFiledValue(Object object,String fieldName) {
        int value = 0;
        try {
            Field field = ImageView.class.getDeclaredField(fieldName);
            field.setAccessible(true);
            int fieldValue = field.getInt(object);
            if(fieldValue > 0 && fieldValue < Integer.MAX_VALUE) {
                value = fieldValue;
            }
        } catch (NoSuchFieldException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (IllegalArgumentException e) {
            e.printStackTrace();
        }
        return value;
    }

    //根据显示的宽和高对图片进行压缩
    private Bitmap decodeSampleBitmapFromPath(String path, int width, int height) {

        //获取图片的宽和高,并不把图片加载到内存中
        BitmapFactory.Options  options = new BitmapFactory.Options();

        options.inJustDecodeBounds = true;
        BitmapFactory.decodeFile(path, options);
        options.inSampleSize = caculateInSampleSize(options, width, height);

        options.inJustDecodeBounds = false; //把图片加载到内存中
        Bitmap bitmap = BitmapFactory.decodeFile(path, options);//已经进行压缩
        return bitmap;
    }

    private int caculateInSampleSize(Options options, int reqWidth, int reqHeight) {
        int width = options.outWidth;
        int height = options.outHeight;
        int inSampleSize = 1;
        if(width > reqWidth || height > reqHeight) {
            int widthRadio = Math.round(1.0f * width / reqWidth);
            int heightRadio = Math.round(1.0f * width / reqHeight);
            inSampleSize = Math.max(widthRadio, heightRadio);
        }
        return inSampleSize;
    }

    private synchronized void addTask(Runnable runnable) {
        mTaskQueue.add(runnable);
//        if(mPoolThreadHandler == null)
//            wait();
        try {
            if(mSemaphorePoolThreadHandler == null)
            mSemaphorePoolThreadHandler.acquire();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        mPoolThreadHandler.sendEmptyMessage(0x110); //发送通知
    }

    private Bitmap getBitmapFromLruCache(String key) {
        return mLruCache.get(key);
    }

    private class ImageSize {
        int width;
        int height;
    }

    private class ImageBeanHolder{ //防止错乱
        Bitmap bitmap;
        ImageView image;
        String path;
    }
}

做自己的软件的Gallery(一)的更多相关文章

  1. Nginx是什么,有什么优点?为什么选择Nginx做web服务器软件?(经典经典)

    1.基础知识 代理服务器:    一般是指局域网内部的机器通过代理服务器发送请求到互联网上的服务器,代理服务器一般作用在客户端.应用比如:GoAgent,FQ神器.    一个完整的代理请求过程为:客 ...

  2. 用RecyclerView做一个小清新的Gallery效果

    一.简介 RecyclerView现在已经是越来越强大,且不说已经被大家用到滚瓜烂熟的代替ListView的基础功能,现在RecyclerView还可以取代ViewPager实现Banner效果,当然 ...

  3. 用RecyclerView做一个小清新的Gallery效果 - Ryan Lee的博客

    一.简介 RecyclerView现在已经是越来越强大,且不说已经被大家用到滚瓜烂熟的代替ListView的基础功能,现在RecyclerView还可以取代ViewPager实现Banner效果,当然 ...

  4. PCAP文件格式分析(做抓包软件之必备)

    转载源:http://blog.csdn.net/anzijin/article/details/2008333 http://www.ebnd.cn/2009/09/07/file-format-a ...

  5. C#做上位机软件——绘图并传输给下位机

    拿到任务之后首先分成了几个部分: 1.绘图.学习了GDI+ 2.图片保存. 3.将图片转换成byte[].由于使用Socket通信,只能传输byte[]数据,所以这一步是向下位机传输的关键. 相应地, ...

  6. Surfer 软件做等值线图

    使用surfer软件做等值线图 Surfer软件美国Golden Software公司编制的一款以画三维图(等高线,image map,3d surface)的软件. Surfer具有的强大插值功能和 ...

  7. 为什么要做一款ERP软件——开源软件诞生7

    技术之外的探讨--第7篇 用日志记录“开源软件”的诞生 赤龙ERP开源地址: 点亮星标,感谢支持,与开发者交流 kzca2000 码云:https://gitee.com/redragon/redra ...

  8. 我的敏捷、需求分析、UML、软件设计电子书 - 下载(持续更新中)

    我将所有我的电子书汇总在一起,方便大家下载!(持续更新) 文档保存在我的网站——软件知识原创基地上(www.umlonline.org),请放心下载. 1)软件设计是怎样炼成的?(2014-4-1 发 ...

  9. ubuntu一些基本软件安装方法

    ubuntu一些基本软件安装方法 首先说明一下 ubuntu 的软件安装大概有几种方式:1. deb 包的安装方式deb 是 debian 系 Linux 的包管理方式, ubuntu 是属于 deb ...

随机推荐

  1. React Native实现一个自定义模块

    概述 在 前期介绍React Native 项目结构的时候,我们讲解过React的项目组成,其中说过 node_modules 文件夹,这是一个存放 node 模块的地方.我们知道React是用npm ...

  2. Android优化之ViewPager的懒加载

    转载本博客请注明出处:点击打开链接    http://blog.csdn.net/qq_32059827/article/details/52487794 出于对用户消耗流量的考虑,有必要对view ...

  3. MySQL 存储过程探秘

    关于存储过程的优点,本文不再阐述.这里只是对创建存储过程时可能遇到的问题做一下简单的分析. 必备基础 这里说的基础,是相关于如何创建一个存储过程的. DELIMITER:分隔符,定界符. 作用就是:作 ...

  4. Spark技术内幕之任务调度:从SparkContext开始

    SparkContext是开发Spark应用的入口,它负责和整个集群的交互,包括创建RDD,accumulators and broadcast variables.理解Spark的架构,需要从这个入 ...

  5. UNIX网络编程——心跳包

    所谓的心跳包就是在客户端和服务器端间定时通知对方自己状态的一个自己定义的命令字,按照一定的时间间隔发送,类似于心跳,所以叫做心跳包. 一般是用来判断对方(设备,进程或其它网元)是否正常动行,一般采用定 ...

  6. RH阴性血妇女怀孕注意事项

     RH阴性血的妇女怀孕注意事项,本文主要讲解RH阴性血抗体效价检测. 第一.孕前准备:Rh阴性的妇女怀孕前,需要到血液中心或指定医院作ABO和Rh血型鉴定,并且做一次孕前血液免疫学产前检查(血型抗体检 ...

  7. ffplay for mfc 代码备忘

    之前上传了一个开源播放器工程ffplay for mfc.它将ffmpeg项目中的ffplay播放器(ffplay.c)移植到了VC的环境下,并且使用MFC做了一套界面.它可以完成一个播放器播放视频的 ...

  8. Mybatis接口编程原理分析(三)

    前面两篇博客Mybatis接口编程原理分析(一)和Mybatis接口编程原理分析(二)我们介绍了MapperProxyFactory.MapperProxy和MapperMethod的操作及源码分析, ...

  9. STL:list用法详解

    list容器介绍 相对于vector容器的连续线性空间,list是一个双向链表,它有一个重要性质:插入操作和删除操作都不会造成原有的list迭代器失效,每次插入或删除一个元素就配置或释放一个元素空间. ...

  10. golang:使用timingwheel进行大量ticker的优化

    Ticker 最近的项目用go实现的服务器需要挂载大量的socket连接.如何判断连接是否还存活就是我们需要考虑的一个问题了. 通常情况下面,socket如果被客户端正常close,服务器是能检测到的 ...