Android获取本地相册图片和拍照获取图片的实现方法

需求:从本地相册找图片,或通过调用系统相机拍照得到图片。

容易出错的地方:

1、当我们指定了照片的uri路径,我们就不能通过data.getData();来获取uri,而应该直接拿到uri(用全局变量或者其他方式)然后设置给imageView

imageView.setImageURI(uri);

2、我发现手机前置摄像头拍出来的照片只有几百KB,直接用imageView.setImageURI(uri);没有很大问题,但是后置摄像头拍出来的照片比较大,这个时候使用imageView.setImageURI(uri);就容易出现 out of memory(oom)错误,我们需要先把URI转换为Bitmap,再压缩bitmap,然后通过imageView.setImageBitmap(bitmap);来显示图片。

3、将照片存放到SD卡中后,照片不能立即出现在系统相册中,因此我们需要发送广播去提醒相册更新照片。

4、这里用到了sharepreference,要注意用完之后移除缓存。

代码:

MainActivity:

package com.sctu.edu.test;

import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;

import com.sctu.edu.test.tools.ImageTools;

import java.io.File;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date;

public class MainActivity extends AppCompatActivity {

  private static final int PHOTO_FROM_GALLERY = 1;
  private static final int PHOTO_FROM_CAMERA = 2;
  private ImageView imageView;
  private File appDir;
  private Uri uriForCamera;
  private Date date;
  private String str = "";
  private SharePreference sharePreference;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    //Android不推荐使用全局变量,我在这里使用了sharePreference
    sharePreference = SharePreference.getInstance(this);
    imageView = (ImageView) findViewById(R.id.imageView);
  }

  //从相册取图片
  public void gallery(View view) {
    Intent intent = new Intent();
    intent.setType("image/*");
    intent.setAction(Intent.ACTION_GET_CONTENT);
    startActivityForResult(intent, PHOTO_FROM_GALLERY);
  }

  //拍照取图片
  public void camera(View view) {
    Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

    uriForCamera = Uri.fromFile(createImageStoragePath());
    sharePreference.setCache("uri", String.valueOf(uriForCamera));

    /**
     * 指定了uri路径,startActivityForResult不返回intent,
     * 所以在onActivityResult()中不能通过data.getData()获取到uri;
     */
    intent.putExtra(MediaStore.EXTRA_OUTPUT, uriForCamera);
    startActivityForResult(intent, PHOTO_FROM_CAMERA);
  }

  @Override
  protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    //第一层switch
    switch (requestCode) {
      case PHOTO_FROM_GALLERY:
        //第二层switch
        switch (resultCode) {
          case RESULT_OK:
            if (data != null) {
              Uri uri = data.getData();
              imageView.setImageURI(uri);
            }
            break;
          case RESULT_CANCELED:
            break;
        }
        break;
      case PHOTO_FROM_CAMERA:
        if (resultCode == RESULT_OK) {
          Uri uri = Uri.parse(sharePreference.getString("uri"));
          updateDCIM(uri);
          try {
            //把URI转换为Bitmap,并将bitmap压缩,防止OOM(out of memory)
            Bitmap bitmap = ImageTools.getBitmapFromUri(uri, this);
            imageView.setImageBitmap(bitmap);
          } catch (IOException e) {
            e.printStackTrace();
          }

          removeCache("uri");
        } else {
          Log.e("result", "is not ok" + resultCode);
        }
        break;
      default:
        break;
    }
  }

  /**
   * 设置相片存放路径,先将照片存放到SD卡中,再操作
   *
   * @return
   */
  private File createImageStoragePath() {
    if (hasSdcard()) {
      appDir = new File("/sdcard/testImage/");
      if (!appDir.exists()) {
        appDir.mkdirs();
      }
      SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMddHHmmss");
      date = new Date();
      str = simpleDateFormat.format(date);
      String fileName = str + ".jpg";
      File file = new File(appDir, fileName);
      return file;
    } else {
      Log.e("sd", "is not load");
      return null;
    }
  }

  /**
   * 将照片插入系统相册,提醒相册更新
   *
   * @param uri
   */
  private void updateDCIM(Uri uri) {
    Intent intent = new Intent(Intent.ACTION_MEDIA_SCANNER_SCAN_FILE);
    intent.setData(uri);
    this.sendBroadcast(intent);

    Bitmap bitmap = BitmapFactory.decodeFile(uri.getPath());
    MediaStore.Images.Media.insertImage(getContentResolver(), bitmap, "", "");
  }

  /**
   * 判断SD卡是否可用
   *
   * @return
   */
  private boolean hasSdcard() {
    if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
      return true;
    } else {
      return false;
    }
  }

  /**
   * 移除缓存
   *
   * @param cache
   */
  private void removeCache(String cache) {
    if (sharePreference.ifHaveShare(cache)) {
      sharePreference.removeOneCache(cache);
    } else {
      Log.e("this cache", "is not exist.");
    }
  }

}

ImageTools:

package com.sctu.edu.test.tools;

import android.app.Activity;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;

public class ImageTools {

  /**
   * 通过uri获取图片并进行压缩
   *
   * @param uri
   * @param activity
   * @return
   * @throws IOException
   */
  public static Bitmap getBitmapFromUri(Uri uri, Activity activity) throws IOException {
    InputStream inputStream = activity.getContentResolver().openInputStream(uri);
    BitmapFactory.Options options = new BitmapFactory.Options();
    options.inJustDecodeBounds = true;
    options.inDither = true;
    options.inPreferredConfig = Bitmap.Config.ARGB_8888;
    BitmapFactory.decodeStream(inputStream, null, options);
    inputStream.close();

    int originalWidth = options.outWidth;
    int originalHeight = options.outHeight;
    if (originalWidth == -1 || originalHeight == -1) {
      return null;
    }

    float height = 800f;
    float width = 480f;
    int be = 1; //be=1表示不缩放
    if (originalWidth > originalHeight && originalWidth > width) {
      be = (int) (originalWidth / width);
    } else if (originalWidth < originalHeight && originalHeight > height) {
      be = (int) (originalHeight / height);
    }

    if (be <= 0) {
      be = 1;
    }
    BitmapFactory.Options bitmapOptinos = new BitmapFactory.Options();
    bitmapOptinos.inSampleSize = be;
    bitmapOptinos.inDither = true;
    bitmapOptinos.inPreferredConfig = Bitmap.Config.ARGB_8888;
    inputStream = activity.getContentResolver().openInputStream(uri);

    Bitmap bitmap = BitmapFactory.decodeStream(inputStream, null, bitmapOptinos);
    inputStream.close();

    return compressImage(bitmap);
  }

  /**
   * 质量压缩方法
   *
   * @param bitmap
   * @return
   */
  public static Bitmap compressImage(Bitmap bitmap) {
    ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
    bitmap.compress(Bitmap.CompressFormat.JPEG, 100, byteArrayOutputStream);
    int options = 100;
    while (byteArrayOutputStream.toByteArray().length / 1024 > 100) {
      byteArrayOutputStream.reset();
      //第一个参数 :图片格式 ,第二个参数: 图片质量,100为最高,0为最差 ,第三个参数:保存压缩后的数据的流
      bitmap.compress(Bitmap.CompressFormat.JPEG, options, byteArrayOutputStream);
      options -= 10;
    }
    ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());
    Bitmap bitmapImage = BitmapFactory.decodeStream(byteArrayInputStream, null, null);
    return bitmapImage;
  }
}

AndroidMainfest.xml:

<?xml version="1.0" encoding="utf-8"?>
<manifest package="com.sctu.edu.test"
   xmlns:android="http://schemas.android.com/apk/res/android">
 <uses-feature
  android:name="android.hardware.camera"
  android:required="true"
  />

 <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
 <uses-permission android:name="android.permission.CAMERA"/>
 <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
 <uses-permission android:name="com.miui.whetstone.permission.ACCESS_PROVIDER"/>
 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
 <uses-feature android:name="android.hardware.camera.autofocus" />

 <application
  android:allowBackup="true"
  android:icon="@mipmap/ic_launcher"
  android:label="@string/app_name"
  android:supportsRtl="true"
  android:theme="@style/AppTheme">
  <activity android:name=".MainActivity">
   <intent-filter>
    <action android:name="android.intent.action.MAIN"/>

    <category android:name="android.intent.category.LAUNCHER"/>
   </intent-filter>
  </activity>
 </application>

</manifest>

activity_main.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
 xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:background="#fff"
 android:orientation="vertical"
 tools:context="com.sctu.edu.test.MainActivity">

 <Button
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:text="从图库找图片"
  android:id="@+id/gallery"
  android:onClick="gallery"
  android:background="#ccc"
  android:textSize="20sp"
  android:padding="10dp"
  android:layout_marginLeft="30dp"
  android:layout_marginTop="40dp"
  />

 <Button
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:text="拍照获取图片"
  android:id="@+id/camera"
  android:onClick="camera"
  android:background="#ccc"
  android:textSize="20sp"
  android:padding="10dp"
  android:layout_marginLeft="30dp"
  android:layout_marginTop="40dp"
  />

 <ImageView
  android:layout_width="300dp"
  android:layout_height="300dp"
  android:id="@+id/imageView"
  android:scaleType="fitXY"
  android:background="@mipmap/ic_launcher"
  android:layout_marginTop="40dp"
  android:layout_marginLeft="30dp"
  />

</LinearLayout>

效果图:

 

或许有人会问,在Android6.0上面怎么点击拍照就出现闪退,那是因为我设置的最高SDK版本大于23,而我现在还没对运行时权限做处理,也许我会在下一篇博客里处理这个问题。谢谢浏览,希望对你有帮助!

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

时间: 2017-03-12

Android 7.0中拍照和图片裁剪适配的问题详解

前言 Android 7.0系统发布后,拿到能升级的nexus 6P,就开始了7.0的适配.发现在Android 7.0以上,在相机拍照和图片裁剪上,可能会碰到以下一些错误: Process: com.yuyh.imgsel, PID: 22995 // 错误1 android.os.FileUriExposedException: file:///storage/emulated/0/Android/data/com.yuyh.imgsel/cache/1486438962645.jpg ex

Android自定义组件获取本地图片和相机拍照图片

iOS中有封装好的选择图片后长按出现动画删除效果,效果如下 而Android找了很久都没有找到有这样效果的第三方组件,最后懒得找了还是自己实现这效果吧 选择图片后还可对图片进行剪裁 当然,代码中还有很多不完善的地方,我接下来会继续完善这个组件的 已经上传到开源社区,欢迎大家来Star啊~ Demo源码:传送门 设计中的碰到的一些问题和解决思路 1.如何让加号图片显示在GridView最后面 首先在调用GridAdapter构造方法时就加载加号图片 /** * 图片适配器 * @param con

Android通过手机拍照或从本地相册选取图片设置头像

像微信.QQ.微博等社交类的APP,通常都有设置头像的功能,设置头像通常有两种方式: 1.让用户通过选择本地相册之类的图片库中已有的图像,裁剪后作为头像. 2.让用户启动手机的相机拍照,拍完照片后裁剪,然后作为头像. 我现在写一个简单的完整代码例子,说明如何在android中实现上述两个头像设置功能. MainActivity.Java文件: package zhangpgil.photo; import java.io.File; import android.support.v7.app.A

Android 通过Intent调用系统拍照程序出现图片太小的问题解决办法

Android 通过Intent调用系统拍照程序出现图片太小的问题解决办法 Intent it = newIntent("android.media.action.IMAGE_CAPTURE"); startActivityForResult(it, Activity.DEFAULT_KEYS_DIALER); 按下拍照键后,会返回到你的activity,所以你的activity要在onActivityResult方法里加一个处理, protectedvoidonActivityRes

Android Camera实现毫秒级拍照实例

我们知道自定义Camera需要以下几步 打开相机,即实例化Camera对象,Camera camera = Camera.open(); 设置Camera的相关参数,Camera.Parameters parameters = camera.getParameters(); 打开预览,camera.setPreviewDisplay(surfaceholder); camera.startPreview(); 获取图片,这里只是从预览中获取因此使用,camera.setPreviewCallb

Android自定义相机实现定时拍照功能

这篇博客为大家介绍Android自定义相机,并且实现倒计时拍照功能. 首先自定义拍照会用到SurfaceView控件显示照片的预览区域,以下是布局文件: activity_main.xml <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="m

Android仿微信单击拍照长按录像功能实例代码

此文章是看郭神公众号发的一篇,仅作学习. 在modlue gradle中添加 compile 'cjt.library.wheel:camera:0.0.7' 在project gradle中添加 compile 'cjt.library.wheel:camera:0.0.7' 添加的地方是 allprojects { repositories { jcenter() /*在此处添加*/ } } 使用起来很方便,只需在xml布局中 <com.cjt2325.cameralibrary.JCame

Android判断是否有拍照权限的实例代码

下面一段代码给大家介绍android判断是否有拍照权限,具体代码如下所示: /** * 返回true 表示可以使用 返回false表示不可以使用 */ public boolean cameraIsCanUse() { boolean isCanUse = true; Camera mCamera = null; try { mCamera = Camera.open(); Camera.Parameters mParameters = mCamera.getParameters(); //针对

Android部分手机拍照后获取的图片被旋转问题的解决方法

调用Android系统拍照功能后,三星手机拍摄后的照片被旋转了90度,横着拍给你变成竖的,竖的拍给你变成横的.其它品牌的手机都是正常的,就三星出现这个怪事. 在Android适配上,我原来一直以为国内的小米手机够奇葩了,结果还有更奇葩的!你说你没事旋转照片干啥,实在是猜不透其居心何在,纯粹是在给开发者制造麻烦啊! 解决办法是获取到拍照后照片被旋转的角度,再旋转回去就好了. 具体思路: 1.首先在调用拍照方法时,保存拍照后的相片原图,得到原图路径,(PhotoBitmapUtils是我自己写的一个

解决android有的手机拍照后上传图片被旋转的问题

需求:做仿新浪发微博的项目,能够上传图片还有两外一个项目用到手机拍摄图片,这两个都需要把图片上传到服务器 遇到问题:有的手机拍摄的图片旋转90度,有的图片旋转了180度,有的手机是正常的,服务器要求的是正的,这样问题就来了,不能用户发个照片在微博上看到的是被旋转了的啊,另外一个项目里旋转了的图片直接匹配出现问题,这个更严重. 解决:开始的时候在网上没有找到很好的解决办法,谷歌百度的搜了一通,想到第一种解决方式,当手机拍照结束,在返回结果处理里面立即跳转到一个新的页面,在新的页面里让用户自己手动去

Android应用中拍照后获取照片路径并上传的实例分享

Activity 中的代码,我只贴出重要的事件部分代码 public void doPhoto(View view) { destoryBimap(); String state = Environment.getExternalStorageState(); if (state.equals(Environment.MEDIA_MOUNTED)) { Intent intent = new Intent("android.media.action.IMAGE_CAPTURE"); s

php更新mysql后获取影响的行数发生异常解决方法

从manual上知道了mysql_affected_rows函数当UPDATE前后的数据一样时会返回异常值, 下面有个方便的解决办法,从官方munual上看到 bdobrica at gmail dot com 留言的: As a solution to the problem pointed in the post reffering to mysql_affected_rows() returning 0 when you are making an update query and the

Android手机拍照或选取图库图片作为头像

package zhangpgil.photo; import java.io.File; import android.support.v7.app.ActionBarActivity; import android.view.View; import android.widget.Button; import android.widget.ImageView; import android.widget.Toast; import android.content.Intent; import

Android圆形头像拍照后“无法加载此图片”的问题解决方法(适配Android7.0)

Feature: 点击选择拍照或者打开相册,选取图片进行裁剪最后设置为圆形头像. Problem: 拍好照片,点击裁剪,弹Toast"无法加载此图片". Solution: 在裁剪的class里加两行代码 intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); intent.addFlags(Intent.FLAG_GRANT_WRITE_URI_PERMISSION); 主要代码如下: public static final S

Android调用手机拍照功能的方法

本文实例讲述了Android调用手机拍照功能的方法.分享给大家供大家参考.具体如下: 一.main.xml布局文件: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" andr

Android 录制手机屏幕视频生成GIF图片实例详解

Android 录制手机屏幕视频生成GIF图片实例详解 无图无真相,在我们日常的网络交流中往往需要给交流对象提供直观的显示,而视频是一个很好的方式,但是视频需要播放器,还需要当做文件进行对点传输,并不是很方便.想CSDN这样的博客网站也并不支持在博客里放视频这种方式,除非你贴外链,太烦了不是么.最好是如下图这种gif方式,直观 今天来教大家一个易操作的录制方式.当然,一般只适合Android开发者.因为你需要有AndroidStudio 工具 AndroidStudio(完成手机屏幕的视频录制,

Android百度地图定位后获取周边位置的实现代码

本文实例讲解Android百度地图定位后获取周边位置的实现代码,分享给大家供大家参考,具体内容如下 效果图: 具体代码: 1.布局文件 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical&q