Android 从相册中选择照片并返回

Android 从相册中选择照片并返回

本项目由某大神的技术博客修改而来,具体的内容可以转入此博客细看:

  • 调用相册处理之后返回到所要显示的imageview

细则

参考自《第一行代码》及相关资料,这里存一遍后以后就不用再重头打一遍了。。闭嘴

首先由于进行读写操作,要在 AndroidManifest.xml中声明权限:

<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />  

调用系统相册:

private static final int CHOOSE_PHOTO=0;  
Intent intent = new Intent("android.intent.action.GET_CONTENT");  
intent.setType("image/*");  
startActivityForResult(intent, CHOOSE_PHOTO);  

然后回调:

@Override  
   protected void onActivityResult(int requestCode, int resultCode, Intent data) {  
       switch (requestCode) {  
           case CHOOSE_PHOTO:  
               if (resultCode == RESULT_OK) {  
                   Bitmap bitmap = null;  
                   //判断手机系统版本号  
                   if (Build.VERSION.SDK_INT >= 19) {  
                       //4.4及以上系统使用这个方法处理图片  
                       bitmap = ImgUtil.handleImageOnKitKat(this, data);        //ImgUtil是自己实现的一个工具类  
                   } else {  
                       //4.4以下系统使用这个方法处理图片  
                       bitmap = ImgUtil.handleImageBeforeKitKat(this, data);  
                   }  
                   ImageView view = (ImageView) findViewById(R.id.personal_info_header_img);  
                   view.setImageBitmap(bitmap);  
               }  
               break;  
           default:  
               break;  
       }  
   }  

将对图像的相关操作封装成一个ImgUtil类,便于使用:

import android.annotation.TargetApi;  
import android.content.ContentUris;  
import android.content.Context;  
import android.content.Intent;  
import android.content.SharedPreferences;  
import android.database.Cursor;  
import android.graphics.Bitmap;  
import android.graphics.BitmapFactory;  
import android.net.Uri;  
import android.preference.PreferenceManager;  
import android.provider.DocumentsContract;  
import android.provider.MediaStore;  
import android.text.TextUtils;  

import java.io.ByteArrayInputStream;  
import java.io.ByteArrayOutputStream;  

/** 
 * Created by wbin on 2016/3/22. 
 */  
public class ImgUtil {  

    //4.4及以上系统使用这个方法处理图片  
    @TargetApi(19)  
    public static Bitmap handleImageOnKitKat(Context context, Intent data) {  
        String imagePath = null;  
        Uri uri = data.getData();  
        if (DocumentsContract.isDocumentUri(context, uri)) {  
            //如果是document类型的Uri,则通过document id处理  
            String docId = DocumentsContract.getDocumentId(uri);  
            if ("com.android.providers.media.documents".equals(uri.getAuthority())) {  
                String id = docId.split(":")[1];  //解析出数字格式的id  
                String selection = MediaStore.Images.Media._ID + "=" + id;  
                imagePath = getImagePath(context, MediaStore.Images.Media.EXTERNAL_CONTENT_URI, selection);  
            } else if ("com.android.providers.downloads.documents".equals(uri.getAuthority())) {  
                Uri contentUri = ContentUris.withAppendedId(  
                        Uri.parse("content://downloads/public_downloads"), Long.valueOf(docId));  
                imagePath = getImagePath(context, contentUri, null);  
            }  
        } else if ("content".equalsIgnoreCase(uri.getScheme())) {  
            //如果不是document类型的Uri,则使用普通方式处理  
            imagePath = getImagePath(context, uri, null);  
        }  
        return getImage(imagePath);  
    }  

    //4.4以下系统使用这个方法处理图片  
    public static Bitmap handleImageBeforeKitKat(Context context, Intent data) {  
        Uri uri = data.getData();  
        String imagePath = getImagePath(context, uri, null);  
        return getImage(imagePath);  
    }  

    public static String getImagePath(Context context, Uri uri, String selection) {  
        String path = null;  
        //通过Uri和selection来获取真实的图片路径  
        Cursor cursor = context.getContentResolver().query(uri, null, selection, null, null);  
        if (cursor != null) {  
            if (cursor.moveToFirst()) {  
                path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));  
            }  
            cursor.close();  
        }  
        return path;  
    }  

    //对bitmap进行质量压缩  
    public static Bitmap compressImage(Bitmap image) {  
        ByteArrayOutputStream baos = new ByteArrayOutputStream();  
        image.compress(Bitmap.CompressFormat.JPEG, 100, baos);//质量压缩方法,这里100表示不压缩,把压缩后的数据存放到baos中  
        int options = 100;  
        while (baos.toByteArray().length / 1024 > 100) {    //循环判断如果压缩后图片是否大于100kb,大于继续压缩  
            baos.reset();//重置baos即清空baos  
            image.compress(Bitmap.CompressFormat.JPEG, options, baos);//这里压缩options%,把压缩后的数据存放到baos中  
            options -= 10;//每次都减少10  
        }  
        ByteArrayInputStream isBm = new ByteArrayInputStream(baos.toByteArray());//把压缩后的数据baos存放到ByteArrayInputStream中  
        Bitmap bitmap = BitmapFactory.decodeStream(isBm, null, null);//把ByteArrayInputStream数据生成图片  
        return bitmap;  
    }  

    //传入图片路径,返回压缩后的bitmap  
    public static Bitmap getImage(String srcPath) {  
        if (TextUtils.isEmpty(srcPath))  //如果图片路径为空 直接返回  
            return null;  
        BitmapFactory.Options newOpts = new BitmapFactory.Options();  
        //开始读入图片,此时把options.inJustDecodeBounds 设回true了  
        newOpts.inJustDecodeBounds = true;  
        Bitmap bitmap = BitmapFactory.decodeFile(srcPath, newOpts);//此时返回bm为空  

        newOpts.inJustDecodeBounds = false;  
        int w = newOpts.outWidth;  
        int h = newOpts.outHeight;  
        //现在主流手机比较多是800*480分辨率,所以高和宽我们设置为  
        float hh = 800f;//这里设置高度为800f  
        float ww = 480f;//这里设置宽度为480f  
        //缩放比。由于是固定比例缩放,只用高或者宽其中一个数据进行计算即可  
        int be = 1;//be=1表示不缩放  
        if (w > h && w > ww) {//如果宽度大的话根据宽度固定大小缩放  
            be = (int) (newOpts.outWidth / ww);  
        } else if (w < h && h > hh) {//如果高度高的话根据宽度固定大小缩放  
            be = (int) (newOpts.outHeight / hh);  
        }  
        if (be <= 0)  
            be = 1;  
        newOpts.inSampleSize = be;//设置缩放比例  
        //重新读入图片,注意此时已经把options.inJustDecodeBounds 设回false了  
        bitmap = BitmapFactory.decodeFile(srcPath, newOpts);  
        return compressImage(bitmap);//压缩好比例大小后再进行质量压缩  
    }  
}  

为了兼容新老版本的手机,我们做了一个判断,如果是4.4及以上系统的手机就调用handleImageOnKitKat()方法来处理图片,否则就调用handleImageBeforeKitKat()方法来处理图片。之所以要这么做,是因为Android系统从4.4版本开始,选取相册的图片不再返回图片真是的Uri了,而是一个封装过的Uri,因此如果是4.4版本以上的手机需要对这个Uri进行解析才行。

当然了,获取到图片路径后不推荐直接使用 BitmapFactory.decodeFile(imgPath)来获取bitmap,因为某些图片体积可能很大,直接加载到内存中有可能会导致程序崩溃(我就遇到过了..你可以直接加载手机高像素拍的原图片试试看=。=)。 所以更好的做法是先对图片进行适当的压缩,然后再加载到内存中(上述代码中实现了)。

好了,以后需要直接来这复制就行了,不用再蛋疼打这么多了=。= 懒人福利。


  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,以下是实现从相册选择照片并显示的Android代码: 1. 在Manifest文件添加读取外部存储权限: ```xml <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> ``` 2. 在布局文件添加一个ImageView用于显示选的图片: ```xml <ImageView android:id="@+id/imageView" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_margin="16dp" android:src="@drawable/ic_launcher_background" /> ``` 3. 在Activity或Fragment添加以下代码实现从相册选择照片并显示: ```java // 声明一个用于返回结果的常量 private static final int REQUEST_CODE_PICK_IMAGE = 1001; // 点击按钮选择照片 findViewById(R.id.buttonPickImage).setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { // 创建Intent对象,设置类型为image/* Intent intent = new Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI); intent.setType("image/*"); // 启动Intent,选择照片 startActivityForResult(Intent.createChooser(intent, "选择照片"), REQUEST_CODE_PICK_IMAGE); } }); // 处理选择照片的结果 @Override protected void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode == REQUEST_CODE_PICK_IMAGE && resultCode == RESULT_OK && data != null) { // 获取选照片的URI Uri uri = data.getData(); // 将URI转换成Bitmap并显示在ImageView try { Bitmap bitmap = MediaStore.Images.Media.getBitmap(getContentResolver(), uri); imageView.setImageBitmap(bitmap); } catch (IOException e) { e.printStackTrace(); } } } ``` 以上代码,我们首先声明了一个常量用于返回结果,然后在点击按钮时创建了一个Intent对象,设置类型为image/*,启动Intent,选择照片。最后在处理选择照片的结果时,获取选照片的URI,将URI转换成Bitmap并显示在ImageView

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值