android自带的处理Bitmap out Memory 的处理,第三方开源的那个更方便,自己练习的话还是很好的

/**
 * @author [email protected]
 * @time 20140606
 */
package com.intbird.utils;

import java.lang.ref.WeakReference;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Bitmap.Config;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.PorterDuff.Mode;
import android.graphics.PorterDuffXfermode;
import android.graphics.Rect;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.Drawable;
import android.media.ThumbnailUtils;
import android.os.AsyncTask;
import android.widget.ImageView;

public class BitmapHelper {

	private CacheManager cacheManager;

	public static final int IMG_TYPE_RES=0;
	public static final int IMG_TYPE_WEB=1;
	public static final int IMG_TYPE_FILE=3;

	private Bitmap bmpHolder=null;
	private Bitmap bmpNoImg=null;

	public BitmapHelper(Context context,int loadingId,int loadNoId){
		cacheManager=CacheManager.getInstance();
		bmpHolder=BitmapFactory.decodeResource(context.getResources(),loadingId);
		bmpNoImg=BitmapFactory.decodeResource(context.getResources(),loadNoId);
	}

	/**
	 * 加载图片
	 * @param type 网络文件,本地文件,还是资源文件
	 * @param fileUrl url
	 * @param imageView 控件
	 * @param width 要获取的指定高度
	 * @param height 要获取的指定宽度
	 */
	public void commonLoadBitmap(int type,String fileUrl,ImageView imageView,int width,int height){
		//内存和文件中没有图片,重新获取;
		Bitmap bmp=cacheManager.getBitmapFromCache(fileUrl);
		if(bmp!=null){
			imageView.setImageBitmap(bmp);
		}
		else{
			switch(type){
			case IMG_TYPE_WEB:
				loadMultiBitmapFromWeb(fileUrl, imageView,width,height);
				break;
			case IMG_TYPE_FILE:
				imageView.setImageBitmap(getBitmapFormFile(	fileUrl, width, height, true));
				break;
			case IMG_TYPE_RES:
				imageView.setImageResource(Integer.parseInt(fileUrl));
				break;
			}
		}
	}

	/**
	 * 设置ImageView为获取指定文件地址的指定高度指定宽度的图片;
	 * @param fileUrl 文件地址
	 * @param reqWidth 目标宽度
	 * @param reqHeight 目标高度
	 * @return bitmap
	 */
	public Bitmap loadSingleBitmapFromFile(String fileUrl,ImageView iv,int reqWidth,int reqHeight){
		BitmapFactory.Options options=new BitmapFactory.Options();
		options.inJustDecodeBounds=true;
		options.inSampleSize=calculateInSampleSize(options, reqWidth, reqHeight);
		BitmapFactory.decodeFile(fileUrl,options);
		options.inJustDecodeBounds=false;
		Bitmap bmp=BitmapFactory.decodeFile(fileUrl,options);
		if(iv!=null) iv.setImageBitmap(bmp);
		return 	bmp;
	}
	/**
	 * 设置ImageView为需要加载的一张网络图片;
	 * @param webUrl
	 * @param imageView
	 * @param width
	 * @param heigth
	 */
	public void loadSingleBitmapFromWeb(String webUrl,ImageView imageView,int width,int heigth){
		BitmapHelper.BitmapWorkerSingleTask task=new BitmapWorkerSingleTask(imageView);
		task.execute(webUrl,width+"",heigth+"");
	}

	/**
	 * adapter中加载图片
	 * @param fileUrl 图片地址
	 * @param imageView 图片控件
	 * @param width 要得到的宽度
	 * @param height 要的到的高度
	 */
	public void loadMultiBitmapFromWeb(String fileUrl, ImageView imageView,int width,int height) {

	    if (cancelPotentialWork(fileUrl, imageView)) {

	        final BitmapMultiWorkerTask task = new BitmapMultiWorkerTask(imageView);

	        final AsyncDrawable asyncDrawable =new AsyncDrawable(bmpHolder, task);

	        imageView.setImageDrawable(asyncDrawable);

	        task.execute(fileUrl,width+"",height+"");
	    }
	}

	/**
	 * 从网络中加载一个需要的Bitmap;
	 * @param webUrl
	 * @param reqWidth
	 * @param reqHeight
	 * @return
	 */
	public Bitmap getBitmapFormWeb(String webUrl,int reqWidth,int reqHeight){
		BitmapFactory.Options options=new BitmapFactory.Options();
		options.inJustDecodeBounds=true;
		options.inSampleSize=calculateInSampleSize(options, reqWidth, reqHeight);
		return ConnInternet.loadBitmapFromNet(webUrl, options);
	}

	/**
	 * 将图片文件转换成bitmap
	 *
	 * @param fileUrl
	 *            图片文件路径
	 * @param width
	 *            宽度
	 * @param height
	 *            高度
	 * @param isThumbnail
	 *            是否根据高宽生成缩略图
	 * @return
	 */
	public Bitmap getBitmapFormFile(String fileUrl, int width, int height,
			boolean isThumbnail) {
		Bitmap bitmap=loadSingleBitmapFromFile(fileUrl,null, width, height);
		// 生成固定尺寸的缩略图
		if (isThumbnail) {
			bitmap = ThumbnailUtils.extractThumbnail(bitmap, width, height,
					ThumbnailUtils.OPTIONS_RECYCLE_INPUT);
		}
		cacheManager.addBitmapToCache(fileUrl, bitmap);
		return bitmap;
	}

	/**
	 * 转换图片成圆形
	 *
	 * @param bitmap
	 * @return
	 */
	public static Bitmap changeBitmapToRound(Bitmap bitmap) {
		if(bitmap==null)
			return null;
		int width = bitmap.getWidth();
		int height = bitmap.getHeight();
		float roundPx;
		float left, top, right, bottom, dst_left, dst_top, dst_right, dst_bottom;
		if (width <= height) {
			roundPx = width / 2;
			left = 0;
			top = 0;
			right = width;
			bottom = width;
			height = width;
			dst_left = 0;
			dst_top = 0;
			dst_right = width;
			dst_bottom = width;
		} else {
			roundPx = height / 2;
			float clip = (width - height) / 2;
			left = clip;
			right = width - clip;
			top = 0;
			bottom = height;
			width = height;
			dst_left = 0;
			dst_top = 0;
			dst_right = height;
			dst_bottom = height;
		}

		Bitmap output = Bitmap.createBitmap(width, height, Config.ARGB_8888);
		Canvas canvas = new Canvas(output);

		final int color = 0xff424242;
		final Paint paint = new Paint();
		final Rect src = new Rect((int) left, (int) top, (int) right,
				(int) bottom);
		final Rect dst = new Rect((int) dst_left, (int) dst_top,
				(int) dst_right, (int) dst_bottom);
		paint.setAntiAlias(true);// 设置画笔无锯齿

		canvas.drawARGB(0, 0, 0, 0); // 填充整个Canvas
		paint.setColor(color);

		// 以下有两种方法画圆,drawRounRect和drawCircle
		// canvas.drawRoundRect(rectF, roundPx, roundPx, paint);//
		// 画圆角矩形,第一个参数为图形显示区域,第二个参数和第三个参数分别是水平圆角半径和垂直圆角半径。
		canvas.drawCircle(roundPx, roundPx, roundPx, paint);

		paint.setXfermode(new PorterDuffXfermode(Mode.SRC_IN));// 设置两张图片相交时的模式,参考http://trylovecatch.iteye.com/blog/1189452
		canvas.drawBitmap(bitmap, src, dst, paint); // 以Mode.SRC_IN模式合并bitmap和已经draw了的Circle
		return output;
	}

	/**
	 * 计算所需大小的图片 的压缩比例
	 * @param options
	 * @param reqWidth 所需宽度
	 * @param reqHeight 所需高度
	 * @return 压缩比例
	 */
	public  int calculateInSampleSize(BitmapFactory.Options options, int reqWidth, int reqHeight) {
	    final int height = options.outHeight;
	    final int width = options.outWidth;
	    int inSampleSize = 1;
	    if (height > reqHeight || width > reqWidth) {
	        final int halfHeight = height / 2;
	        final int halfWidth = width / 2;
	    	while ((halfHeight / inSampleSize) > reqHeight
	            && (halfWidth / inSampleSize) > reqWidth) {
	    		inSampleSize *= 2;
	    	}
		}
		return inSampleSize;
	}

	/**
	 * 检查启动的任务
	 * @param fileUrl 文件路径
	 * @param imageView 目标控件
	 * @return 是否开始加载任务
	 */
	private  boolean cancelPotentialWork(String fileUrl, ImageView imageView) {

	    final BitmapMultiWorkerTask bitmapWorkerTask = getBitmapWorkerTask(imageView);

	    if (bitmapWorkerTask != null) {

	        String bitmapUrl = bitmapWorkerTask.fileUrl;
	        if (bitmapUrl!=null&&bitmapUrl != fileUrl) {

	            bitmapWorkerTask.cancel(true);
	            return true;
	        }

	        else {
	            return false;
	        }
	    }
	    return true;
	}

	/**
	 * 获取该图片控件中的加载任务
	 * @param imageView
	 * @return 返回该任务
	 */
	private  BitmapMultiWorkerTask getBitmapWorkerTask(ImageView imageView) {
		   if (imageView != null) {
		       final Drawable drawable = imageView.getDrawable();
		       if (drawable instanceof AsyncDrawable) {
		           final AsyncDrawable asyncDrawable = (AsyncDrawable) drawable;
		           return asyncDrawable.getBitmapWorkerTask();
		       }
		    }
		    return null;
	}
	private class AsyncDrawable extends BitmapDrawable {
	    private final WeakReference<BitmapMultiWorkerTask> bitmapWorkerTaskReference;
	    @SuppressWarnings("deprecation")
		public AsyncDrawable(Bitmap bitmap,BitmapMultiWorkerTask bitmapWorkerTask) {
	        super(bitmap);
	        bitmapWorkerTaskReference =new WeakReference<BitmapMultiWorkerTask>(bitmapWorkerTask);
	    }
	    public BitmapMultiWorkerTask getBitmapWorkerTask() {
	        return bitmapWorkerTaskReference.get();
	    }
	}
	private class BitmapMultiWorkerTask extends AsyncTask<String, Void, Bitmap> {
		private WeakReference<ImageView> imageViewReference;
		private String fileUrl;
		public BitmapMultiWorkerTask(ImageView imageView){
			imageViewReference=new WeakReference<ImageView>(imageView);
		}
		@Override
		protected Bitmap doInBackground(String... params) {
			fileUrl=params[0];
			int reqWidth=Integer.valueOf(params[1]);
			int reqHeight=Integer.valueOf(params[2]);
			Bitmap bitmap=getBitmapFormWeb(fileUrl,reqWidth,reqHeight);
			return bitmap;
		}

		@Override
	    protected void onPostExecute(Bitmap bitmap) {

	        if (isCancelled()) {
	            bitmap = null;
	        }

	        if (imageViewReference != null) {
	            final ImageView imageView = imageViewReference.get();

	            final BitmapMultiWorkerTask bitmapWorkerTask =getBitmapWorkerTask(imageView);

	            if (this == bitmapWorkerTask && imageView != null) {

	            	if(bitmap!=null){
	            		imageView.setImageBitmap(bitmap);
	            		cacheManager.addBitmapToCache(fileUrl, bitmap);
	            	}

	            	else{
	            		imageView.setImageBitmap(bmpNoImg);
	            	}
	            }
	        }
	    }
	}
	/**
	 * 异步加载一张图片
	 * @author [email protected]
	 *
	 */
	private class BitmapWorkerSingleTask extends AsyncTask<String, Void, Bitmap>{
		private WeakReference<ImageView> imageViewReference;
		private String fileUrl="";
		public BitmapWorkerSingleTask(ImageView imageView){
			imageViewReference=new WeakReference<ImageView>(imageView);
		}
		@Override
		protected Bitmap doInBackground(String... params) {
			fileUrl=params[0];
			int reqWidth=Integer.valueOf(params[1]);
			int reqHeight=Integer.valueOf(params[2]);
			Bitmap bitmap=getBitmapFormWeb(fileUrl,reqWidth,reqHeight);
			return bitmap;
		}
		@Override
		protected void onPostExecute(Bitmap bitmap) {
			 if (imageViewReference != null) {
		            final ImageView imageView = imageViewReference.get();
		            if (imageView != null) {
		            	if(bitmap!=null){
		            		imageView.setImageBitmap(bitmap);
		            	}
	            	   else{
			            	  imageView.setImageBitmap(bmpNoImg);
			            }
		            }
		    }
		}
	}
}

android自带的处理Bitmap out Memory 的处理,第三方开源的那个更方便,自己练习的话还是很好的

时间: 2024-10-10 16:17:09

android自带的处理Bitmap out Memory 的处理,第三方开源的那个更方便,自己练习的话还是很好的的相关文章

android自带的处理Bitmap out Memory 的处理,我仅仅是改变了些写法成为自己用的东西

每天上万次的启动载入证明这个载入是不错的; 第三方的载入框架非常多,推荐使用成熟框架,比方大家都知道的ImageLoad等, 这里的仅仅供学习; 我也曾在一个菜谱的项目里写过载入手机相冊图片的,只是当时写的不好,有空我会再写一个,这个项目里用这个jar包,上万次的载入偶尔看到三星的报两三个OOM还好 缓存管理在这里 http://blog.csdn.net/intbird/article/details/38338713 图片处理在这里 http://blog.csdn.net/intbird/

android 向服务器Get和Post请求的两种方式,android向服务器发送文件,自己组装协议和借助第三方开源

/** * @author [email protected] * @time 20140606 */ package com.intbird.utils; import java.io.BufferedReader; import java.io.DataOutputStream; import java.io.File; import java.io.FileInputStream; import java.io.IOException; import java.io.InputStream

android自带的内存memory和第三方外部存储disk管理

/** * @author [email protected] * @time 20140606 */ package com.intbird.utils; import java.io.File; import com.yilake.store.FileHelper; import android.graphics.Bitmap; import android.os.Environment; import android.util.LruCache; public class CacheMan

android自带的浏览器如何下载apk包

============问题描述============ 如题! 目前android上其它的浏览器因为自己做过处理,所以下载没问题.. 可是android自带的浏览器下载有时候下载错误.有时候下载的是.htm文件,恶心啊,貌似把apk当成恶意软件了,请问大神们该怎么处理.. 和contentType设置有关吗,我换了好多个,没用啊,我的服务端是用java写的 ============解决方案1============ response.setContentType("application/vn

调用android自带的json类解析出错!!!

============问题描述============ {"weatherinfo":{"city":"北京","cityid":"101010100","temp1":"2","temp2":"15","weather":"小到中雨转小雨","img1":"

Android自带音频均衡器MusicFx分析

Android自带音频均衡器MusicFx分析 种种原因,我要简单分析一个Android中built-in的音频均衡器MusicFx.重点是它的默认值的来历.网上很少有文章讲了这个的除了这篇<com.android.musicFx设置音效流程 -- 从app到AudioFlinger>.注:Android系统版本为4.2.2_r1. 从App到AudioFliger的终点是在android_media_AudioEffect.cpp(之前版本在是audio_media_AudioEffect.

android 实现带清除效果的EditText(附带抖动效果)

Android一直没有提供类似于ios中自带清除效果的输入框(ios只要只要添加属性即可实现),所以在Android当中 想要实现此效果就需要使用自定义控件的方式实现. 思路:可以使用一个Linearlayout里面横向布局一个EditText和一个删除的图片,监听输入框的焦点和文字变化,设置图片的显隐和点击清除事件.但是这么做些弊端,首先增加了UI布局的层级结构不利于UI结构的优化而且可能会出现文字过长遮挡住图片的情况.所以采用自定义控件继承于EditText,使用getCompoundDra

android 在HTML中显示bitmap

逻辑:将bitmap转化为Base64,通过调用HTML中的JS,显示到HTML中 (1)android代码 public String bitmaptoString(Bitmap bitmap) { // 将Bitmap转换成Base64字符串 StringBuffer string = new StringBuffer(); ByteArrayOutputStream bStream = new ByteArrayOutputStream(); try { bitmap.compress(C

android自带theme

在网上搜了一下,android自带theme如下: ?android:theme="@android:style/Theme.Dialog"   将一个Activity显示为对话框模式 ?android:theme="@android:style/Theme.NoTitleBar"  不显示应用程序标题栏 ?android:theme="@android:style/Theme.NoTitleBar.Fullscreen"  不显示应用程序标题栏