android手机图库功能的实现,Android 选择相册照片并返回功能的实现代码

本文介绍了在Android中如何从相册选择图片并进行处理的代码,包括权限声明、启动相册选择、回调处理、适配不同系统版本的图片路径获取以及图片的质量压缩。通过ImgUtil工具类,实现了对图片的压缩,避免因图片过大导致内存问题。文章强调了在4.4及以上版本需要特殊处理Uri,以及推荐先压缩图片再加载到内存中的最佳实践。
摘要由CSDN通过智能技术生成

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

调用系统相册:

private static final int CHOOSE_PHOTO=0;

Intent intent = new Intent(“android.intent.action.GET_CONTENT”);

intent.setType(”image/*”);

startActivityForResult(intent, CHOOSE_PHOTO);

[java] view plain copy

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;

}

}

@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);//压缩好比例大小后再进行质量压缩

}

}

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,因为某些图片体积可能很大,直接加载到内存中有可能会导致程序崩溃(我就遇到过了..你可以直接加载手机高像素拍的原图片试试看=。=)。 所以更好的做法是先对图片进行适当的压缩,然后再加载到内存中(上述代码中实现了)。

总结

以上所述是小编给大家介绍的Android 选择相册照片并返回功能的实现代码,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值