在Android中,Intent触发Camera程序,拍好照片后,将会返回数据,可是kao虑到内存问题,Camera不会将全尺寸的图像返回给调用的Activity,通常状况下,有可能返回的是缩略图,好比120*160px。android
这是为何呢?这不是一个Bug,而是通过精心设计的,却对开发者不透明。git
好比摄像头800W像素,根据我目前设置拍出来的图片尺寸为3200*2400px。有人说,那就返回呗,大不了耗1-2M的内存,不错,这个尺寸的图片确实只有1.8M左右的大小。可是你想不到的是,这个尺寸对应的Bitmap会耗光你应用程序的全部内存。Android出于安全性kao虑,只会给你一个寒碜的缩略图。github
在Android2.3中,默认的Bitmap为32位,类型是ARGB_8888,也就意味着一个像素点占用4个字节的内存。咱们来作一个简单的计算题:3200*2400*4 bytes = 30M,因此用拍照获得的图片须要裁剪处理后使用。安全
Android裁剪图片的Intent附加数据的具体含义,拿图说事儿:spa
Intent("com.android.camera.action.CROP")对应的全部可选数据都一目了然。在了解上面个个选项的含义以后,咱们将目光着眼于三个极为重要的选项:.net
data、MediaStore.EXTRA_OUTPUT以及return-data。设计
data和MediaStore.EXTRA_OUTPUT都是可选的传入数据选项,你能够选择设置data为Bitmap,或者将相应的数据与URI关联起来,你也能够选择是否返回数据(return-data: true)。code
为何还有不用返回数据的选项?若是对URI足够了解的话,应该知道URI与File类似,你全部的操做如裁剪将数据都保存在了URI中,你已经持有了相应的URI,也就无需画蛇添足,再返回Bitmap了。orm
前面已经说到,能够设置data为Bitmap,可是这种操做的限制在于,你的Bitmap不能太大。所以,咱们前进的思路彷佛明确了:截大图用URI,小图用Bitmap。图片
我将这个思路整理成一张图片:
在上面,我就拍照截图这一需求进行了详细的分析,试图让你们了解Android自己的限制,以及咱们应当采起的实现方案。
根据咱们的分析与总结,图片的来源有拍照和相册,而可采起的操做有
使用Bitmap并返回数据
使用Uri不返回数据
前面咱们了解到,使用Bitmap有可能会致使图片过大,而不能返回实际大小的图片,我将采用大图Uri,小图Bitmap的数据存储方式。
咱们将要使用到URI来保存拍照后的图片:
private static final String IMAGE_FILE_LOCATION = "file:///sdcard/temp.jpg";//temp fileUri imageUri = Uri.parse(IMAGE_FILE_LOCATION);//The Uri to store the big bitmap
不难知道,咱们从相册选取图片的Action为Intent.ACTION_GET_CONTENT。
根据咱们上一篇博客的分析,我准备好了两个实例的Intent。
1、从相册截大图:
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 detectionstartActivityForResult(intent, CHOOSE_BIG_PICTURE);
2、从相册截小图
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 detectionstartActivityForResult(intent, CHOOSE_SMALL_PICTURE);
3、对应的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进行操做。
1、首先准备好须要使用到的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
2、使用MediaStore.ACTION_IMAGE_CAPTURE能够轻松调用Camera程序进行拍照:
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);//action is captureintent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri); startActivityForResult(intent, TAKE_BIG_PICTURE);//or TAKE_SMALL_PICTURE
3、接下来就能够在 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); }
4、最后一步,咱们已经将数据传入裁剪图片程序,接下来要作的就是处理返回的数据了:
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; }
摘选自:https://github.com/ryanhoo/PhotoCropper