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,而我现在还没对运行时权限做处理,也许我会在下一篇博客里处理这个问题。谢谢浏览,希望对你有帮助!

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

(0)

相关推荐

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

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

  • 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 Camera实现毫秒级拍照实例

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

  • 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 通过Intent调用系统拍照程序出现图片太小的问题解决办法

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

  • 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通过手机拍照或从本地相册选取图片设置头像

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

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

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

  • Android图片或拍照选择图片功能实例代码

    前言 一般公司都有更换用户头像功能,需要从图库中选择图片或者拍照,基本还会对图片进行裁剪.最近抽空就做了一些简单的封装,方便以后使用.主要是用了建造者模式,链式调用,方便简单.可以自定义图片路径,附带裁剪和简单压缩功能.使用实例如下: ChooseImageTask.getInstance() .createBuilder(this) .setFileName("图片名称")//有默认的 .setFilePath("图片路径")//有默认的 .setIsCrop(f

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

    需求:从本地相册找图片,或通过调用系统相机拍照得到图片. 容易出错的地方: 1.当我们指定了照片的uri路径,我们就不能通过data.getData();来获取uri,而应该直接拿到uri(用全局变量或者其他方式)然后设置给imageView imageView.setImageURI(uri); 2.我发现手机前置摄像头拍出来的照片只有几百KB,直接用imageView.setImageURI(uri);没有很大问题,但是后置摄像头拍出来的照片比较大,这个时候使用imageView.setIm

  • Git获取本地分支对应的远端服务器分支名的方法

    可以使用下面命令查看本地分支在远端服务器的分支名: $ git rev-parse --abbrev-ref local_branch_name@{upstream} 把 local_branch_name 换成要查询的本地分支名,例如 master 等.下面通过例子来说明这个命令各个参数的含义. 先创建一个新的本地分支,名为 new_local_branch,关连到远端服务器的 Remote_Branch_U 分支: $ git checkout -b new_local_branch aos

  • Android中通过访问本地相册或者相机设置用户头像实例

    目前几乎所有的APP在用户注册时都会有设置头像的需求,大致分为三种情况: (1)通过获取本地相册的图片,经过裁剪后作为头像. (2)通过启动手机相机,现拍图片然后裁剪作为头像. (3)在APP中添加一些自带的头像资源,供用户选择(不够人性化,目前很少使用). 这次我们简单介绍下通过获取本地相册以及相机拍摄的方法设置头像,实现思路如下: (1)通过startActivityForResult方法,分别传递调用系统相册的Intent和调用相机拍照的Intent来做选择 (2)调用Android系统中

  • Android拍照裁剪图片

    下面是效果图,看看是不是亲想要的效果图,如果是,这段代码你就可以参考下了,但是要灵活运用,根据需求做相应的改动. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation

  • vue里的axios如何获取本地json数据

    目录 使用axios获取本地json数据 使用axios获取本地json文件,报404错 解决方案:main.js里 使用axios获取本地json数据 第一步:在已有的vue项目中安装axios命令 npm install --save axios 第二步:在main.js里导入下面的代码 import axios from 'axios' 第三步:在static文件夹下添加testData文件夹,在这个文件夹下添加data.json文件 第四步:在需要数据的页面引入axios,异步加载数据

  • android图像绘制(六)获取本地图片或拍照图片等图片资源

    从SD卡中获取图片资源,或者拍一张新的图片. 先贴代码 获取图片: 注释:拍照获取的话,可以指定图片的保存地址,在此不说明. 复制代码 代码如下: CharSequence[] items = {"相册", "相机"}; new AlertDialog.Builder(this) .setTitle("选择图片来源") .setItems(items, new OnClickListener() { public void onClick(Dia

  • Android 拍照选择图片并上传功能的实现思路(包含权限动态获取)

    作为一个Android新手,想实现手机拍照并上传的功能,经过查找资料,已实现此功能.在此记录备忘.老鸟请忽略. 一.实现思路: 1.Android手机客户端,拍照(或选择图片),然后上传到服务器. 2.服务器端接收手机端上传上来的图片. 二.实现步骤: 1.按惯例,先放效果图: 项目结构: 2.activity_main.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:a

随机推荐