文档章节

Android大图片裁剪解决方案

今日竹石
 今日竹石
发布于 2014/04/15 11:42
字数 1787
阅读 379
收藏 14
点赞 1
评论 0

截图原理

在Android中,Intent触发Camera程序,拍好照片后,将会返回数据,但是考虑到内存问题,Camera不会将全尺寸的图像返回给调用的Activity,一般情况下,有可能返回的是缩略图,比如120*160px。

    这是为什么呢?这不是一个Bug,而是经过精心设计的,却对开发者不透明。

    以我的小米手机为例,摄像头800W像素,根据我目前设置拍出来的图片尺寸为3200*2400px。有人说,那就返回呗,大不了耗1-2M的内存,不错,这个尺寸的图片确实只有1.8M左右的大小。但是你想不到的是,这个尺寸对应的Bitmap会耗光你应用程序的所有内存。Android出于安全性考虑,只会给你一个寒碜的缩略图。

    在Android2.3中,默认的Bitmap为32位,类型是ARGB_8888,也就意味着一个像素点占用4个字节的内存。我们来做一个简单的计算题:3200*2400*4 bytes =   30M。

    如此惊人的数字!哪怕你愿意为一张生命周期超不过10s的位图愿意耗费这么巨大的内存,Android也不会答应的。

1 Mobile devices typically have constrained system resources.
2 Android devices can have as little as 16MB of memory available to a single application.

    这是Android Doc的原文,虽然不同手机系统的厂商可能围绕16M这个数字有微微的上调,但是这30M,一般的手机还真挥霍不起。也只有小米这种牛机,内存堪比个人PC,本着土财主般挥金如土的霸气才能做到。

    OK,说了这么多,无非是吐吐苦水,爆爆个人经历而已,实际的解决方案在哪里呢?

    我也是Google到的,话说一般百度不了的问题,那就Google或者直接StackOverFlow,只不过得看英文罢了。

    最后翻来覆去,我在国外的一个Android团队的博客中找到了相应的方案,印证了我的猜想同时也给出了实际的代码。

    我将这篇文章翻译成了中文,作为本博客的基础,建议详细看看。

【译】如何使用Android MediaStore裁剪大图片

    这篇博客了不起的地方在于解决了Android对返回图片的大小限制,并且详细解释了裁剪图片的Intent附加数据的具体含义。OK,我只是站在巨人的肩膀上,改善方案,适应更广泛需求而已。

    拿图说事儿:

    Intent("com.android.camera.action.CROP")对应的所有可选数据都一目了然。在了解上面个个选项的含义之后,我们将目光着眼于三个极为重要的选项:

    dataMediaStore.EXTRA_OUTPUT以及return-data

    data和MediaStore.EXTRA_OUTPUT都是可选的传入数据选项,你可以选择设置data为Bitmap,或者将相应的数据与URI关联起来,你也可以选择是否返回数据(return-data: true)。

    为什么还有不用返回数据的选项?如果对URI足够了解的话,应该知道URI与File相似,你所有的操作如裁剪将数据都保存在了URI中,你已经持有了相应的URI,也就无需多此一举,再返回Bitmap了。

    前面已经说到,可以设置data为Bitmap,但是这种操作的限制在于,你的Bitmap不能太大。因此,我们前进的思路似乎明确了:截大图用URI,小图用Bitmap。

    我将这个思路整理成一张图片:

从相册截图

    根据我们的分析与总结,图片的来源有拍照和相册,而可采取的操作有

  • 使用Bitmap并返回数据

  • 使用Uri不返回数据

    前面我们了解到,使用Bitmap有可能会导致图片过大,而不能返回实际大小的图片,我将采用大图Uri,小图Bitmap的数据存储方式。

    我们将要使用到URI来保存拍照后的图片:

1 private static final String IMAGE_FILE_LOCATION = "file:///sdcard/temp.jpg";//temp file
2 Uri imageUri = Uri.parse(IMAGE_FILE_LOCATION);//The Uri to store the big bitmap

    不难知道,我们从相册选取图片的Action为Intent.ACTION_GET_CONTENT。

    根据我们上一篇博客的分析,我准备好了两个实例的Intent。

    一、从相册截大图:

01 Intent intent = new Intent(Intent.ACTION_GET_CONTENT, null);
02 intent.setType("image/*");
03 intent.putExtra("crop""true");
04 intent.putExtra("aspectX"2);
05 intent.putExtra("aspectY"1);
06 intent.putExtra("outputX"600);
07 intent.putExtra("outputY"300);
08 intent.putExtra("scale"true);
09 intent.putExtra("return-data"false);
10 intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
11 intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
12 intent.putExtra("noFaceDetection"true); // no face detection
13 startActivityForResult(intent, CHOOSE_BIG_PICTURE);

    二、从相册截小图

01 Intent intent = new Intent(Intent.ACTION_GET_CONTENT, null);
02 intent.setType("image/*");
03 intent.putExtra("crop""true");
04 intent.putExtra("aspectX"2);
05 intent.putExtra("aspectY"1);
06 intent.putExtra("outputX"200);
07 intent.putExtra("outputY"100);
08 intent.putExtra("scale"true);
09 intent.putExtra("return-data"true);
10 intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
11 intent.putExtra("noFaceDetection"true); // no face detection
12 startActivityForResult(intent, CHOOSE_SMALL_PICTURE);

    三、对应的onActivityResult可以这样处理返回的数据

01 switch (requestCode) {
02 case CHOOSE_BIG_PICTURE:
03     Log.d(TAG, "CHOOSE_BIG_PICTURE: data = " + data);//it seems to be null
04     if(imageUri != null){
05         Bitmap bitmap = decodeUriAsBitmap(imageUri);//decode bitmap
06         imageView.setImageBitmap(bitmap);
07     }
08     break;
09 case CHOOSE_SMALL_PICTURE:
10     if(data != null){
11         Bitmap bitmap = data.getParcelableExtra("data");
12         imageView.setImageBitmap(bitmap);
13     }else{
14         Log.e(TAG, "CHOOSE_SMALL_PICTURE: data = " + data);
15     }
16     break;
17 default:
18     break;
19 }
01 private Bitmap decodeUriAsBitmap(Uri uri){
02     Bitmap bitmap = null;
03     try {
04         bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(uri));
05     catch (FileNotFoundException e) {
06         e.printStackTrace();
07         return null;
08     }
09     return bitmap;
10 }

    效果图:

大图 小图


拍照截图

拍照截图有点儿特殊,要知道,现在的Android智能手机的摄像头都是几百万的像素,拍出来的图片都是非常大的。因此,我们不能像对待相册截图一样使用Bitmap小图,无论大图小图都统一使用Uri进行操作。

    一、首先准备好需要使用到的Uri:

1 private static final String IMAGE_FILE_LOCATION = "file:///sdcard/temp.jpg";//temp file
2 Uri imageUri = Uri.parse(IMAGE_FILE_LOCATION);//The Uri to store the big bitmap

    二、使用MediaStore.ACTION_IMAGE_CAPTURE可以轻松调用Camera程序进行拍照:

1 Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);//action is capture
2 intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
3 startActivityForResult(intent, TAKE_BIG_PICTURE);//or TAKE_SMALL_PICTURE

    三、接下来就可以在 onActivityResult中拿到返回的数据(Uri),并将Uri传递给截图的程序。

01 switch (requestCode) {
02 case TAKE_BIG_PICTURE:
03     Log.d(TAG, "TAKE_BIG_PICTURE: data = " + data);//it seems to be null
04     //TODO sent to crop
05     cropImageUri(imageUri, 800400, CROP_BIG_PICTURE);
06
07     break;
08 case TAKE_SMALL_PICTURE:
09     Log.i(TAG, "TAKE_SMALL_PICTURE: data = " + data);
10     //TODO sent to crop
11     cropImageUri(imageUri, 300150, CROP_SMALL_PICTURE);
12
13     break;
14 default:
15     break;
16 }

    可以看到,无论是拍大图片还是小图片,都是使用的Uri,只是尺寸不同而已。我们将这个操作封装在一个方法里面。

01 private void cropImageUri(Uri uri, int outputX, int outputY, int requestCode){
02     Intent intent = new Intent("com.android.camera.action.CROP");
03     intent.setDataAndType(uri, "image/*");
04     intent.putExtra("crop""true");
05     intent.putExtra("aspectX"2);
06     intent.putExtra("aspectY"1);
07     intent.putExtra("outputX", outputX);
08     intent.putExtra("outputY", outputY);
09     intent.putExtra("scale"true);
10     intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
11     intent.putExtra("return-data"false);
12     intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
13     intent.putExtra("noFaceDetection"true); // no face detection
14     startActivityForResult(intent, requestCode);
15 }

    四、最后一步,我们已经将数据传入裁剪图片程序,接下来要做的就是处理返回的数据了:

01 switch (requestCode) {
02 case CROP_BIG_PICTURE://from crop_big_picture
03     Log.d(TAG, "CROP_BIG_PICTURE: data = " + data);//it seems to be null
04     if(imageUri != null){
05         Bitmap bitmap = decodeUriAsBitmap(imageUri);
06         imageView.setImageBitmap(bitmap);
07     }
08     break;
09 case CROP_SMALL_PICTURE:
10     if(imageUri != null){
11         Bitmap bitmap = decodeUriAsBitmap(imageUri);
12         imageView.setImageBitmap(bitmap);
13     }else{
14         Log.e(TAG, "CROP_SMALL_PICTURE: data = " + data);
15     }
16     break;
17 default:
18     break;
19 }

效果图:








© 著作权归作者所有

共有 人打赏支持
今日竹石
粉丝 41
博文 227
码字总数 181312
作品 0
朝阳
程序员
Android大图片裁剪终极解决方案(中:从相册截图)

转载声明:Ryan的博客文章欢迎您的转载,但在转载的同时,请注明文章的来源出处,不胜感激! :-) http://my.oschina.net/ryanhoo/blog/86853 在这篇博客中,我将向大家展示如何从相册截图。 ...

RyanHoo
2012/11/03
0
14
Android大图片裁剪终极解决方案(下:拍照截图)

转载声明:Ryan的博客文章欢迎您的转载,但在转载的同时,请注明文章的来源出处,不胜感激! :-) http://my.oschina.net/ryanhoo/blog/86865 上一篇博客中,我们学习到了如何使用Android相册...

RyanHoo
2012/11/03
0
11
Android大图片裁剪终极解决方案(上:原理分析)

转载声明:Ryan的博客文章欢迎您的转载,但在转载的同时,请注明文章的来源出处,不胜感激! :-) http://my.oschina.net/ryanhoo/blog/86842 约几个月前,我正为公司的APP在Android手机上实现...

RyanHoo
2012/11/03
0
11
Android大图片裁剪终极解决方案(上:原理分析)

约几个月前,我正为公司的APP在Android手机上实现拍照截图而烦恼不已。 上网搜索,确实有不少的例子,大多都是抄来抄去,而且水平多半处于demo的样子,可以用来讲解知识点,但是一碰到实际项...

小克898
2012/11/12
0
1
Android大图片裁剪终极解决方案(下:拍照截图)

上一篇博客中,我们学习到了如何使用Android相册截图。在这篇博客中,我将向大家展示如何拍照截图。 拍照截图有点儿特殊,要知道,现在的Android智能手机的摄像头都是几百万的像素,拍出来的...

小克898
2012/11/12
0
4
android根据屏幕宽度,按比例缩放图片

ImageView有scaleType属性可以缩放图片,让图片铺满屏幕宽度,但是会出现压缩或裁剪的情况。 ImageView的scaleType的属性分别是matrix(默认)、center、centerCrop、centerInside、fitCent...

penngo
2015/07/22
0
0
系统接口调用-关于调用图库并裁剪图片

一. 实现代码 调用系统相册 代码: Intent intent = new Intent(Intent.ACTIONGETCONTENT, null);intent.setType("image/*");intent.putExtra("crop", "true");intent.putExtra("aspectX", 2......

liqiuzuo
2015/01/20
0
3
【译】如何使用Android MediaStore裁剪大图片

转载声明:Ryan的博客文章欢迎您的转载,但在转载的同时,请注明文章的来源出处,不胜感激! :-) http://my.oschina.net/ryanhoo/blog/86843 译者:Ryan Hoo 来源:http://www.androidworks...

RyanHoo
2012/11/03
0
2
Android平台上裁剪m4a

Android手机上设置铃声的操作是比较灵活的,一般读者听到一首喜欢的歌曲,马上就可以对这首歌曲进行裁剪,裁剪到片段后,再通过系统的接口设置为铃声(电话铃声、闹钟铃声等)。 前提是,播放...

奇哥3
04/28
0
0
Android 性能优化:手把手教你优化Bitmap图片资源的使用

前言 在 开发中,性能优化策略十分重要 本文主要讲解性能优化中的Bitmap 使用优化,希望你们会喜欢 目录 1. 优化原因 即 为什么要优化图片资源,具体如下图:

Carson_Ho
04/24
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

Kafka设计解析(一)- Kafka背景及架构介绍

原创文章,转载请务必将下面这段话置于文章开头处。(已授权InfoQ中文站发布) 本文转发自技术世界,原文链接 http://www.jasongj.com/2015/03/10/KafkaColumn1 摘要   Kafka是由LinkedI...

mskk
3分钟前
0
0
使用Service Mesh整合您的微服务架构

在微服务架构的世界中,它正在达到这样的程度,即管理系统的复杂性对于利用它带来的好处变得至关重要。 目前,如何实现这些微服务不再是一个问题,因为有很多可用的框架(Spring Boot,Vert....

xiaomin0322
6分钟前
0
0
看看 LinkedList Java 9

终于迎来了 LinkedList 类,实现的接口就有点多了 Serializable, Cloneable, Iterable<E>, Collection<E>, Deque<E>, List<E>, Queue<E>。LinkedList是一个实现了List接口和Deque接口的双端链......

woshixin
24分钟前
0
0
算法 - 冒泡排序 C++

大家好,我是ChungZH。今天我给大家讲一下最基础的排序算法:冒泡排序(BubbleSort)。 冒泡排序算法的原理如下: 比较相邻的元素。如果第一个比第二个大(可以相反),就交换他们两个。 对每...

ChungZH
27分钟前
0
0
jquery ajax request payload和fromData请求方式

请求头的不同 fromData var data = { name : 'yiifaa'};// 提交数据$.ajax('app/', { method:'POST', // 将数据编码为表单模式 contentType:'application/x-ww...

lsy999
29分钟前
0
0
阿里P7架构师,带你点亮程序员蜕变之路

前言: Java是现阶段中国互联网公司中,覆盖度最广的研发语言。 掌握了Java技术体系,不管在成熟的大公司,快速发展的公司,还是创业阶段的公司,都能有立足之地。 有不少朋友问,成为Java架...

Java大蜗牛
31分钟前
1
0
Ecstore 在没有后台管理界面(维护)的情况如何更新表的字段

window 系统: 切换到:app\base 目录下: C:\Users\qimh>d: D:\>cd D:\WWW\huaqh\app\base 执行:D:\WWW\huaqh\app\base>cmd update linux 系统: 1># cd /alidata/www.novoeshop.com/app/......

qimh
35分钟前
0
0
设计模式-策略模式

策略模式 解释 对工厂模式的再次封装,使用参数控制上下文信息(将工厂返回的实例赋值给context field) 不会返回bean实例,只是设置对应的条件 调用context的方法(调用field的方法) 用户只...

郭里奥
38分钟前
0
0
python使用有序字典

python自带的collections包中有很多有用的数据结构可供使用,其中有个叫OrderedDict类,它可以在使用的时候记录元素插入顺序,在遍历使用的时候就可以按照原顺序遍历。 a = {"a":1,"b"...

芝麻糖人
今天
0
0
RestTemplate HttpMessageConverter

RestTemplate 微信接口 text/plain HttpMessageConverter

微小宝
今天
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部