截圖原理android
在Android中,Intent觸發Camera程序,拍好照片後,將會返回數據,可是考慮到內存問題,Camera不會將全尺寸的圖像返回給調用的Activity,通常狀況下,有可能返回的是縮略圖,好比120*160px。安全
這是爲何呢?這不是一個Bug,而是通過精心設計的,卻對開發者不透明。app
以個人小米手機爲例,攝像頭800W像素,根據我目前設置拍出來的圖片尺寸爲3200*2400px。有人說,那就返回唄,大不了耗1-2M的內存,不錯,這個尺寸的圖片確實只有1.8M左右的大小。可是你想不到的是,這個尺寸對應的Bitmap會耗光你應用程序的全部內存。Android出於安全性考慮,只會給你一個寒磣的縮略圖。spa
在Android2.3中,默認的Bitmap爲32位,類型是ARGB_8888,也就意味着一個像素點佔用4個字節的內存。咱們來作一個簡單的計算題:3200*2400*4 bytes = 30M。.net
如此驚人的數字!哪怕你願意爲一張生命週期超不過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,說了這麼多,無非是吐吐苦水,爆爆我的經歷而已,實際的解決方案在哪裏呢?code
我也是Google到的,話說通常百度不了的問題,那就Google或者直接StackOverFlow,只不過得看英文罷了。orm
最後翻來覆去,我在國外的一個Android團隊的博客中找到了相應的方案,印證了個人猜測同時也給出了實際的代碼。blog
我將這篇文章翻譯成了中文,做爲本博客的基礎,建議詳細看看。
【譯】如何使用Android MediaStore裁剪大圖片
這篇博客了不得的地方在於解決了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來保存拍照後的圖片:
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。
1、從相冊截大圖:
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); |
2、從相冊截小圖
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); |
3、對應的onActivityResult能夠這樣處理返回的數據
01 |
switch (requestCode) { |
02 |
case CHOOSE_BIG_PICTURE: |
03 |
Log.d(TAG, "CHOOSE_BIG_PICTURE: data = " + data); //it seems to be null |
05 |
Bitmap bitmap = decodeUriAsBitmap(imageUri); //decode bitmap |
06 |
imageView.setImageBitmap(bitmap); |
09 |
case CHOOSE_SMALL_PICTURE: |
11 |
Bitmap bitmap = data.getParcelableExtra( "data" ); |
12 |
imageView.setImageBitmap(bitmap); |
14 |
Log.e(TAG, "CHOOSE_SMALL_PICTURE: data = " + data); |
01 |
private Bitmap decodeUriAsBitmap(Uri uri){ |
04 |
bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(uri)); |
05 |
} catch (FileNotFoundException e) { |
效果圖:
大圖 |
小圖 |
|
|
拍照截圖
拍照截圖有點兒特殊,要知道,如今的Android智能手機的攝像頭都是幾百萬的像素,拍出來的圖片都是很是大的。所以,咱們不能像對待相冊截圖同樣使用Bitmap小圖,不管大圖小圖都統一使用Uri進行操做。
1、首先準備好須要使用到的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 |
2、使用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 |
3、接下來就能夠在 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 |
05 |
cropImageUri(imageUri, 800 , 400 , CROP_BIG_PICTURE); |
08 |
case TAKE_SMALL_PICTURE: |
09 |
Log.i(TAG, "TAKE_SMALL_PICTURE: data = " + data); |
11 |
cropImageUri(imageUri, 300 , 150 , CROP_SMALL_PICTURE); |
能夠看到,不管是拍大圖片仍是小圖片,都是使用的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); |
4、最後一步,咱們已經將數據傳入裁剪圖片程序,接下來要作的就是處理返回的數據了:
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 |
05 |
Bitmap bitmap = decodeUriAsBitmap(imageUri); |
06 |
imageView.setImageBitmap(bitmap); |
09 |
case CROP_SMALL_PICTURE: |
11 |
Bitmap bitmap = decodeUriAsBitmap(imageUri); |
12 |
imageView.setImageBitmap(bitmap); |
14 |
Log.e(TAG, "CROP_SMALL_PICTURE: data = " + data); |
效果圖: