首先由于进行读写操作,要在 AndroidManifest.xml中声明权限: uses-permission android:name=“android.permission.READ_EXTERNAL_STORAGE” / uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" / 调用系统
首先由于进行读写操作,要在 AndroidManifest.xml中声明权限:
<uses-permission android:name=“android.permission.READ_EXTERNAL_STORAGE” /> <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); [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 选择相册照片并返回功能的实现代码,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对自由互联网站的支持!