`
stephen830
  • 浏览: 3010249 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Android大图片裁剪终极解决方案

 
阅读更多

 

Android大图片裁剪终极解决方案

 

约几个月前,我正为公司的APP在Android手机上实现拍照截图而烦恼不已。

上网搜索,确实有不少的例子,大多都是抄来抄去,而且水平多半处于demo的样子,可以用来讲解知识点,但是一碰到实际项目,就漏洞百出。

当时我用大众化的解决方案,暂时性的做了一个拍照截图的功能,似乎看起来很不错。问题随之而来,我用的是小米手机,在别的手机上都运行正常,小米这里却总是碰钉子。虽然我是个理性的米粉,但是也暗地里把小米的工程师问候了个遍。真是惭愧!

翻文档也找不出个答案来,我一直对com.android.camera.action.CROP持有大大的疑问,它是从哪里来,它能干什么,它接收处理什么类型的数据?Google对此却讳莫如深,在官方文档中只有Intent中有只言片语言及,却不甚详尽。

随着项目的驱动,我不能抱着不了解原理就不往前走的心态,唯一要做的,是解决问题。最后在德问上找到一条解决方案,说是哪怕是大米也没问题。当时乐呵呵将代码改了改,确实在所有的手机上跑起来了,一时如释重负,对这个的疑问也抛诸脑后了。

直到月前,BOSS要求将拍照上传到服务器的图片分辨率加倍。OK,加倍简单,增加outputX以及outputY不就得了?

intent.putExtra("outputX", outputX);
intent.putExtra("outputY", outputY);

这一增加,吓了我一跳。BOSS的手机拍到的照片几乎就是个缩略图,但是被我问候了全体工程师的小米在这个时候就体现出国产神机的范儿了,小米上的尺寸一切正常。这个为什么呢?我大致了解原因,却不知道如何解决。

在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也不会答应的。

Mobile devices typically have constrained system resources.
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裁剪大图片 http://www.linuxidc.com/Linux/2012-11/73939.htm

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

拿图说事儿:



 

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

data、MediaStore.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来保存拍照后的图片:

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

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

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

一、从相册截大图:

Intent intent = new Intent(Intent.ACTION_GET_CONTENT, null);
intent.setType("image/*");
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 2);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 600);
intent.putExtra("outputY", 300);
intent.putExtra("scale", true);
intent.putExtra("return-data", false);
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); // no face detection
startActivityForResult(intent, CHOOSE_BIG_PICTURE);

二、从相册截小图

Intent intent = new Intent(Intent.ACTION_GET_CONTENT, null);
intent.setType("image/*");
intent.putExtra("crop", "true");
intent.putExtra("aspectX", 2);
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 200);
intent.putExtra("outputY", 100);
intent.putExtra("scale", true);
intent.putExtra("return-data", true);
intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
intent.putExtra("noFaceDetection", true); // no face detection
startActivityForResult(intent, CHOOSE_SMALL_PICTURE);

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

switch (requestCode) {
case CHOOSE_BIG_PICTURE:
 Log.d(TAG, "CHOOSE_BIG_PICTURE: data = " + data);//it seems to be null
 if(imageUri != null){
  Bitmap bitmap = decodeUriAsBitmap(imageUri);//decode bitmap
  imageView.setImageBitmap(bitmap);
 }
 break;
case CHOOSE_SMALL_PICTURE:
 if(data != null){
  Bitmap bitmap = data.getParcelableExtra("data");
  imageView.setImageBitmap(bitmap);
 }else{
  Log.e(TAG, "CHOOSE_SMALL_PICTURE: data = " + data);
 }
 break;
default:
 break;
}

 

private Bitmap decodeUriAsBitmap(Uri uri){
 Bitmap bitmap = null;
 try {
  bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(uri));
 } catch (FileNotFoundException e) {
  e.printStackTrace();
  return null;
 }
 return bitmap;
}

 

 

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

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

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

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

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

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

switch (requestCode) {
case TAKE_BIG_PICTURE:
 Log.d(TAG, "TAKE_BIG_PICTURE: data = " + data);//it seems to be null
 //TODO sent to crop
 cropImageUri(imageUri, 800, 400, CROP_BIG_PICTURE);
 
 break;
case TAKE_SMALL_PICTURE:
 Log.i(TAG, "TAKE_SMALL_PICTURE: data = " + data);
 //TODO sent to crop 
 cropImageUri(imageUri, 300, 150, CROP_SMALL_PICTURE);
 
 break;
default:
 break;
}

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

private void cropImageUri(Uri uri, int outputX, int outputY, int requestCode){
 Intent intent = new Intent("com.android.camera.action.CROP");
 intent.setDataAndType(uri, "image/*");
 intent.putExtra("crop", "true");
 intent.putExtra("aspectX", 2);
 intent.putExtra("aspectY", 1);
 intent.putExtra("outputX", outputX);
 intent.putExtra("outputY", outputY);
 intent.putExtra("scale", true);
 intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
 intent.putExtra("return-data", false);
 intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
 intent.putExtra("noFaceDetection", true); // no face detection
 startActivityForResult(intent, requestCode);
}

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

switch (requestCode) {
case CROP_BIG_PICTURE://from crop_big_picture
 Log.d(TAG, "CROP_BIG_PICTURE: data = " + data);//it seems to be null
 if(imageUri != null){
  Bitmap bitmap = decodeUriAsBitmap(imageUri);
  imageView.setImageBitmap(bitmap);
 }
 break;
case CROP_SMALL_PICTURE:
 if(imageUri != null){
  Bitmap bitmap = decodeUriAsBitmap(imageUri);
  imageView.setImageBitmap(bitmap);
 }else{
  Log.e(TAG, "CROP_SMALL_PICTURE: data = " + data);
 }
 break;
default:
 break;
}

 

 

 

 

 

 

 

  • 大小: 19.4 KB
  • 大小: 21.8 KB
分享到:
评论

相关推荐

    Android裁剪图片终极解决方案

    本文将深入探讨如何实现Android大图片裁剪的终极解决方案,结合提供的资源包`CropImage-Enhance`,我们将涵盖以下几个关键知识点: 1. **Bitmap的加载与内存管理** - Android中的Bitmap对象占用大量内存,处理不当...

    android图片裁剪上传头像

    本文将深入探讨如何实现“android图片裁剪上传头像”的功能,包括图片裁剪工具的选择、实现原理以及上传过程。 首先,我们要解决的是图片裁剪问题。在描述中提到的“这是我找到的最好的图片裁剪工具”,很可能是...

    利用编译过的libjpeg库实现Android图片压缩的终极方案

    在Android开发中,图片压缩是一项常见的任务,尤其是在处理大量图片或者优化...在实际开发中,结合项目需求,可以进一步扩展和优化这个解决方案,例如添加图片格式转换、图片裁剪等功能,打造一个全面的图像处理工具。

    android实现超高仿微信终极图片选择器

    "android实现超高仿微信终极图片选择器"项目就是一个致力于满足这一需求的开源解决方案。下面我们将深入探讨这个项目的各个关键知识点。 1. **图片选择器的设计** - **多模式选择**:该选择器支持单选和多选两种...

    jQuery.photoClip手机图片裁剪插件.zip

    jQuery.photoClip是一款专为手机端设计的图片裁剪插件,它提供了便捷且灵活的图片裁剪解决方案,适用于各种Web应用和移动网站。 jQuery.photoClip的核心特性: 1. **跨平台兼容性**:作为基于jQuery的插件,jQuery...

    Android-Frisson提供时髦的图片剪裁功能

    "Android-Frisson提供时髦的图片剪裁功能"这个项目就是针对这一需求而设计的,它为Android开发者提供了一种现代化、用户友好的图片裁剪解决方案。Frisson旨在使图片裁剪操作变得更加直观和吸引人,提升用户体验。 ...

    uCrop:Android图像裁剪库

    implementation 'com.github.yalantis:ucrop:2.2.6'轻量级通用解决方案 implementation 'com.github.yalantis:ucrop:2.2.6-native'获取本机代码以保持图像质量的能力(将apk大小增加约1.5 MB) 将UCropActivity添加...

    完美封装的图片剪切for android

    "完美封装的图片剪切 for Android" 是一个专为Android设计的源码库,它提供了高效且用户体验良好的图片裁剪解决方案。这个库的核心优势在于它的高度封装和内置的动画效果,使得开发者可以直接调用而无需花费大量时间...

    Android中ImageCropper矩形、圆形 裁剪框的实现方法

    在Android开发中,ImageCropper库常常用于图片裁剪功能,提供矩形和圆形裁剪框的选择。本文将深入探讨如何在Android应用中实现这两种裁剪框的详细步骤。 首先,我们关注圆形裁剪框的实现。圆形裁剪框的需求是在用户...

    uCrop - Image Cropping Library for Android

    uCrop 是由 Yalantis 团队开发的开源项目,其主要目标是为Android开发者提供一套强大、易于定制且用户体验优良的图片裁剪解决方案。该库支持多种裁剪模式、比例以及自定义界面元素,使用户可以根据自己的需求调整...

    Croppy:Android图像裁剪库

    总的来说,`Croppy`是一个强大的Android图像裁剪解决方案,它简化了Android应用中的图像裁剪功能的实现,同时提供了丰富的定制选项和良好的用户体验。如果你的项目需要这样的功能,`Croppy`是一个值得考虑的选择。...

    调用系统的相机,图库以及对相片进行剪切的demo

    这是一个常用的Android图片裁剪库,它提供了自定义裁剪比例、旋转、边界限制等功能。开发者可以设置裁剪框的初始位置和大小,裁剪后的图片尺寸,以及是否允许旋转等参数。用户在界面上操作裁剪框后,最终的裁剪结果...

    ImageLoaderUtil

    4. **图片裁剪与缩放**:ImageLoaderUtil 可以根据需要对图片进行裁剪和缩放,避免加载过大图片导致内存溢出,同时可以根据设备屏幕尺寸自动调整图片大小,提高加载效率。 5. **占位符与加载失败图**:在图片加载...

    车牌识别 android 代码 原需要节分11 这个只需要5积分

    "原来的11积分"可能指的是一个完整的解决方案,可能是他人贡献的项目,需要11个积分来获取。这个项目可能包含了一个训练好的模型,用于识别车牌上的字符。在Android平台上,我们可以使用TensorFlow Lite这样的轻量级...

    CombineBitmap.zip

    博客附件通常会包含详细实现过程,包括关键代码片段和可能遇到的问题及解决方案。如果你有这个项目的博客链接,通过阅读博客可以获取更深入的理解和技术细节。对于Android开发者来说,掌握这种组合头像的技巧不仅能...

    SelectCamera-master:两种相机相册截取正方形图片窗口-方式1:系统相机+相册弹窗选择->自定义图片剪裁->圆形ImageView显示,包含图片存储,base64上载,自定义控件注解,imageloader方式2 :弹窗显示图片快照+相机选择->系统图片剪裁->查看详情弹窗->上传(未实现)等。经过真机测试,5.0版本一下使用正常,6.0以上使用会有各种问题(例如:后置相机个数没做处理,但是框架模式很好,欢迎使用者自己更新)推荐使用https

    总之,"SelectCamera-master"是一个实用的Android相机与相册集成解决方案,它集成了多种功能并考虑到了实际应用场景中的常见需求。尽管存在一些兼容性问题,但其开放源代码的特点使得开发者可以对其进行修改和完善,...

    CustomCamera.zip

    10. **错误处理和回调**:良好的错误处理机制是必不可少的,如相机无法打开、存储空间不足等情况,都需要提供相应的提示和解决方案。 以上就是创建一个类似"CustomCamera.zip"中描述的自定义相机应用的关键知识点。...

    网上报名图像压缩工具

    网上报名图像压缩工具是一种专为应对各类网上报名过程中的照片尺寸和大小...总之,网上报名图像压缩工具是解决网络报名照片问题的有效解决方案,帮助用户轻松应对各种报名系统的图片要求,确保顺利上传并完成报名流程。

    ffmpeg视频解码成YUV,JPG

    综上所述,FFmpeg提供了一套完整的解决方案,用于从视频解码到YUV,再到JPEG的处理,并允许对生成的JPEG文件进行各种定制化操作。通过灵活运用FFmpeg的命令行参数和滤镜,我们可以高效地处理多媒体数据,满足各种...

    基于ARM+Linux的车载系统.pptx

    - **解决方案**:通过NMEA协议解析GPS数据,结合陀螺仪进行姿态校正,最终通过短信模块发送。 3. **汽车UI界面开发**: - **难点**:如何在有限的硬件资源下,设计出既美观又实用的操作界面。 - **解决方案**:...

Global site tag (gtag.js) - Google Analytics