Android中使用Camera类编写手机拍照App的实例教程

Camera是Android摄像头硬件的相机类,位于硬件包"android.hardware.Camera"下。它主要用于摄像头捕获图片、启动/停止预览图片、拍照、获取视频帧等,它是设备本地的服务,负责管理设备上的摄像头硬件。

Camera既然用于管理设备上的摄像头硬件,那么它也为开发人员提供了相应的方法,并且这些方法大部分都是native的,用C++在底层实现,下面简单介绍一下Camera的一些方法:

  • static Camera open():打开Camera,返回一个Camera实例。
  • static Camera open(int cameraId):根据cameraId打开一个Camera,返回一个Camera实例。
  • final void release():释放掉Camera的资源。
  • static int getNumberOfCameras():获取当前设备支持的Camera硬件个数。
  • Camera.Parameters getParameters():获取Camera的各项参数设置类。
  • void setParameters(Camera.Parameters params):通过params把Camera的各项参数写入到Camera中。
  • final void setDisplayOrientation(int degrees):摄像预览的旋转度。
  • final void setPreviewDisplay(SurfaceHolder holder):设置Camera预览的SurfaceHolder。
  • final void starPreview():开始Camera的预览。
  • final void stopPreview():停止Camera的预览
  • final void autoFocus(Camera.AutoFocusCallback cb):自动对焦。
  • final takePicture(Camera.ShutterCallback shutter,Camera.PictureCallback raw,Camera.PictureCallback jpeg):拍照。
  • final void lock():锁定Camera硬件,使其他应用无法访问。
  • final void unlock():解锁Camera硬件,使其他应用可以访问。

上面已经介绍了Camera的常用方法,下面根据这些方法详细讲解Android下使用Camera开发拍照应用最基本的过程:

  • 使用open()方法获取一个Camera对象,鉴于Android设备可能配置了多个摄像头,open()方法可以通过摄像头Id开启指定的摄像头。
  • 为Camera对象设置预览类,它是一个SurfaceHolder对象,通过setPreviewDisplay(SurfaceHolder)方法设置。
  • 调用startPreview()方法开始Camera对象的预览。
  • 调用takePicture()方法进行拍照,其中可以通过Camera.PictureCallback()回调获得拍摄的Image数据。
  • 当拍摄完成后,需要调用stopPreview()方法停止预览,并使用release()释放Camera占用的资源。

以上介绍的步骤都是最基本的过程,是必不可少的。Camera没有提供公开的构造函数,只能通过open()方法获取,并且必须设置一个预览类SurfaceHolder,如果不设置的话,将无法使用Camera。在使用完成Camera之后,必须使用release()释放Camera资源。

实例:
使用Camera控制拍照的几个步骤:
 1、调用Camera的open()打开相机
 2、调用Camera的getParameters()获取拍照参数。该方法返回一个Camera.Paremeters对象
 3、调用Camera.Parameters对象方法设置拍照的参数
 4、调用Camera.startPreview()方法开始预览取景,在预览取景之前需要调用Camera的setPreviewDisplay(SurfaceHolder holder)方法设置使用哪个SurfaceView来显示取景图片。
 5、调用Camera的takePicture()方法进行拍照
 6、结束程序时,调用Camera的stopPreview()结束取景预览,并调用release()方法释放资源
 
代码:

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

<RelativeLayout 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"
  tools:context=".MainActivity" > 

  <SurfaceView
    android:id="@+id/sView"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"/>
  <Button
    android:id="@+id/take"
    android:layout_alignParentBottom="true"
    android:onClick="capture"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:text="@string/take"/> 

</RelativeLayout> 

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="match_parent" >
  <EditText
    android:id="@+id/photoNmae"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"/>
  <ImageView
    android:id="@+id/show"
    android:layout_below="@id/photoNmae"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"/> 

</RelativeLayout>
package com.android.xiong.cameratest; 

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException; 

import android.app.Activity;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.graphics.Bitmap;
import android.graphics.Bitmap.CompressFormat;
import android.graphics.BitmapFactory;
import android.graphics.ImageFormat;
import android.hardware.Camera;
import android.hardware.Camera.AutoFocusCallback;
import android.hardware.Camera.PictureCallback;
import android.hardware.Camera.ShutterCallback;
import android.os.Bundle;
import android.os.Environment;
import android.util.DisplayMetrics;
import android.view.Display;
import android.view.Menu;
import android.view.SurfaceHolder;
import android.view.SurfaceHolder.Callback;
import android.view.LayoutInflater;
import android.view.SurfaceView;
import android.view.View;
import android.view.Window;
import android.view.WindowManager;
import android.widget.EditText;
import android.widget.ImageView; 

public class MainActivity extends Activity { 

  SurfaceView sView;
  SurfaceHolder surfaceHodler;
  int screenWidth, screenHeight;
  // 定义系统所用的照相机
  Camera camera;
  // 是否存在预览中
  boolean isPreview = false; 

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    // 设置全屏
    requestWindowFeature(Window.FEATURE_NO_TITLE);
    getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
        WindowManager.LayoutParams.FLAG_FULLSCREEN);
    setContentView(R.layout.activity_main);
    // 获取窗口管理器
    WindowManager wm = getWindowManager();
    Display display = wm.getDefaultDisplay();
    DisplayMetrics metrics = new DisplayMetrics();
    // 获取屏幕的宽和高
    display.getMetrics(metrics);
    screenWidth = metrics.widthPixels;
    screenHeight = metrics.heightPixels;
    sView = (SurfaceView) findViewById(R.id.sView);
    // 设置surface不需要自己的维护缓存区
    sView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
    // 获得SurfaceView的SurfaceHolder
    surfaceHodler = sView.getHolder();
    // 为srfaceHolder添加一个回调监听器
    surfaceHodler.addCallback(new Callback() { 

      @Override
      public void surfaceDestroyed(SurfaceHolder arg0) {
        // 如果camera不为null,释放摄像头
        if (camera != null) {
          if (isPreview)
            camera.stopPreview();
          camera.release();
          camera = null;
        }
      } 

      @Override
      public void surfaceCreated(SurfaceHolder arg0) {
        // 打开摄像头
        initCamera(); 

      } 

      @Override
      public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2,
          int arg3) {
      }
    });
  } 

  private void initCamera() {
    if (!isPreview) {
      // 此处默认打开后置摄像头
      // 通过传入参数可以打开前置摄像头
      camera = Camera.open();
      camera.setDisplayOrientation(90);
    }
    if (!isPreview && camera != null) {
      Camera.Parameters parameters = camera.getParameters();
      // 设置预览照片的大小
      parameters.setPreviewSize(screenWidth, screenHeight);
      // 设置预览照片时每秒显示多少帧的最小值和最大值
      parameters.setPreviewFpsRange(4, 10);
      // 设置照片的格式
      parameters.setPictureFormat(ImageFormat.JPEG);
      // 设置JPG照片的质量
      parameters.set("jpeg-quality", 85);
      // 设置照片的大小
      parameters.setPictureSize(screenWidth, screenHeight);
      // 通过SurfaceView显示取景画面
      try {
        camera.setPreviewDisplay(surfaceHodler);
      } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
      }
      // 开始预览
      camera.startPreview();
      isPreview = true;
    }
  } 

  public void capture(View source) {
    if (camera != null) {
      // 控制摄像头自动对焦后才拍摄
      camera.autoFocus(autoFocusCallback);
    }
  } 

  AutoFocusCallback autoFocusCallback = new AutoFocusCallback() { 

    @Override
    public void onAutoFocus(boolean arg0, Camera arg1) {
      if (arg0) {
        // takePicture()方法需要传入三个监听参数
        // 第一个监听器;当用户按下快门时激发该监听器
        // 第二个监听器;当相机获取原始照片时激发该监听器
        // 第三个监听器;当相机获取JPG照片时激发该监听器
        camera.takePicture(new ShutterCallback() { 

          @Override
          public void onShutter() {
            // 按下快门瞬间会执行此处代码
          }
        }, new PictureCallback() { 

          @Override
          public void onPictureTaken(byte[] arg0, Camera arg1) {
            // 此处代码可以决定是否需要保存原始照片信息
          }
        }, myJpegCallback);
      } 

    }
  };
  PictureCallback myJpegCallback = new PictureCallback() { 

    @Override
    public void onPictureTaken(byte[] data, Camera camera) {
      // 根据拍照所得的数据创建位图
      final Bitmap bm = BitmapFactory.decodeByteArray(data, 0,
          data.length);
      // 加载布局文件
      View saveDialog = getLayoutInflater().inflate(R.layout.save, null);
      final EditText potoName = (EditText) saveDialog
          .findViewById(R.id.photoNmae);
      // 获取saveDialog对话框上的ImageView组件
      ImageView show = (ImageView) saveDialog.findViewById(R.id.show);
      // 显示刚刚拍得的照片
      show.setImageBitmap(bm);
      // 使用AlertDialog组件
      new AlertDialog.Builder(MainActivity.this)
          .setView(saveDialog)
          .setNegativeButton("取消", null)
          .setPositiveButton("保存",
              new DialogInterface.OnClickListener() {
                @Override
                public void onClick(DialogInterface arg0,
                    int arg1) {
                  // 创建一个位于SD卡上的文件
                  File file = new File(Environment
                      .getExternalStorageDirectory()
                      + "/"
                      + potoName.getText().toString()
                      + ".jpg");
                  FileOutputStream fileOutStream=null;
                  try {
                    fileOutStream=new FileOutputStream(file);
                    //把位图输出到指定的文件中
                    bm.compress(CompressFormat.JPEG, 100, fileOutStream);
                    fileOutStream.close();
                  } catch (IOException io) {
                    io.printStackTrace();
                  } 

                }
              }).show();
      //重新浏览
      camera.stopPreview();
      camera.startPreview();
      isPreview=true;
    }
  }; 

  @Override
  public boolean onCreateOptionsMenu(Menu menu) {
    getMenuInflater().inflate(R.menu.main, menu);
    return true;
  } 

}
时间: 2016-04-22

Android变形(Transform)之Camera使用介绍

引言 接Android变形(Transform)之Matrix,来总结下Camera的使用,Camera主要实现3D的变形,有转动,旋转等,Camera的源码是由Native(本地代码)实现,提供的接口也比较简单.官方的介绍:A camera instance can be used to compute 3D transformations and generate a matrix that can be applied, for instance, on a  Canvas. 效果图 原图

Android实现Camera2预览和拍照效果

简介 网上对于 Camera2 的介绍有很多,在 Github 上也有很多关于 Camera2 的封装库,但是对于那些库,封装性太强,有时候我们仅仅是需要个简简单单的拍照功能而已,因此,自定义一个 Camera 使之变得轻量级那是非常重要的了.(本文并非重复造轮子, 而是在于学习 Camera2API 的基本功能, 笔记之.) 学习要点: 使用 Android Camera2 API 的基本功能. 迭代连接到设备的所有相机的特征. 显示相机预览和拍摄照片. Camera2 API 为连接到 An

Android Camera变焦编程步骤

1.添加Camera权限 2.判断是否支持变焦 复制代码 代码如下: public boolean isSupportZoom()     {         boolean isSuppport = true;         if (mCamera.getParameters().isSmoothZoomSupported())         {             isSuppport = false;         }         return isSuppport;    

Android Camera开发手电筒功能

这是一个简单的运用Android Camera开发手电筒功能,AndroidManifest.xml文件的入口是startapp,这个文件没上传上来,大家可以自己写. flashlight.java package com.android.app; import android.app.Activity; import android.hardware.Camera; import android.hardware.Camera.Parameters; import android.os.Bun

Android自定义照相机Camera出现黑屏的解决方法

本文实例讲述了Android自定义照相机Camera出现黑屏的解决方法.分享给大家供大家参考,具体如下: 对于一些手机,像HTC,当自定义Camera时,调用Camera.Parameters的 parameters.setPreviewSize(width, height)方法时,如果width和height为奇数情况下,则会出现黑屏现象,解决办法可参考SDK提供的ApiDemos中关于Camera的 例子: List<Size> sizes = parameters.getSupporte

Android编程中调用Camera时预览画面有旋转问题的解决方法

本文实例讲述了Android编程中调用Camera时预览画面有旋转问题的解决方法.分享给大家供大家参考,具体如下: 在调用Camera写应用的时候,前后摄像头的情况有时候是不一样的.有时候,明明后摄像头没有问题,而调用到前摄像头时,却倒转了180°,或者其他角度,百思不得其解.在查看了Android源码之后,发现它的解决办法很是好,接下来贴个源码,以备日后查看. public static int getDisplayRotation(Activity activity) { int rotat

Android camera实时预览 实时处理,人脸识别示例

Android camera实时预览 实时处理,面部认证. 预览操作是网友共享的代码,我在继承SurfaceView 的CameraSurfaceView 中加入了帧监听事件,每次预览监听前五个数据帧,在处理做一个面部识别. 先看目录关系 自定义控件CameraSurfaceView.java 自定义接口方法CameraInterface.java CameraActivity预览界面. CameraSurfaceView.Java package com.centaur.camera.prev

Android Camera是否支持变焦的判断方法总结

最近老大交给了一个任务,说是要在本地视频端能够调节摄像头焦距. 碰到了一些问题: 1.手机支不支持摄像头变焦 2.系统自带摄像软件可以变焦,但是自己编写的程序不支持变焦, 这个问题网上也有很多童鞋碰到了: 复制代码 代码如下: public void setZoomIn(){     try{         params = camera.getParameters();         zoomValue +=5;         params.setZoom(zoomValue);    

android之camera用法实例详解

本文实例讲述了android之camera用法.分享给大家供大家参考.具体如下: 1.关于预览横竖差90度的问题 原因分析 经过查证和实验,可以证实:Android提供的SDK(android.hardware.Camera)里大概不能正常的使用竖屏(portrait layout)加载照相机,当用竖屏模式加载照相机时会产生以下情况: ①. 照相机成像左倾90度(倾斜): ②. 照相机成像长宽比例不对(失比). 之所以是"大概",原因是因为可能可以通过一些比较复杂的手段解决.如果以上成

android系统在静音模式下关闭camera拍照声音的方法

话说为了防止偷拍,业内有不成文规定,手机公司在做camera时,点击拍照和录像键的时候,必须要有提示音.因此,google也就非常人性化的将播放拍照声音的函数,放到了cameraService中,防止开发者能开发出不响的camera,从而只要调用拍照函数,一定会响,这是写死在framework中的. 话说这个规定在当今有点不合时宜,这不,今天我收到测试提的一个BUG,说是公司的新需求,要求在静音模式下拍照声音也得取消.这么无耻的需求,也许就在我们中国最大的山寨手机公司才会提到.废话不多说,看看是

详解Linux系统三种模式下的简单命令

i的编辑器: 1.三种模式:底行模式 命令模式 插入模式 命令模式: 1.从命令切换插入模式: i : 光标所在左侧输入 I 光标移动到所在行的最左则    o :光标移动的下一行(新的一行) O:光标移动的上一行(新的一行)    a :光标移动到所在行的右则输入 A:光标移动到所在行的最右则 2.复制:(n代表数字) 行: nyy n>0 单词:nyw n>0 3.粘贴:p 4.撤消到上一步骤:u 5.恢复到上一步: ctrl+r 6.替换: 替换一个字符:r 连续替换    :R 7.定

Android开发实现在Wifi下获取本地IP地址的方法

本文实例讲述了Android开发实现在Wifi下获取本地IP地址的方法.分享给大家供大家参考,具体如下: 代码核心介绍: WifiManager类提供了对设备Wifi功能的管理,包括Wifi开关的打开和关闭,IP地址获取等等,早期的Android版本甚至支持对于IP的设置.在这里仅介绍其简单的一个小应用--获取IP地址. 首先先添加Uses-Permission:CHANGE_WIFI_STATE,WAKE_LOCK和ACCESS_WIFI_STATE.代码如下(AndroidManifest.

Android中关于百度糯米app关闭网页或窗口的方法(99%人不知)

我们公司做了一款使用百度钱包的移动网页支付进行支付的产品,用户通过百度钱包.百度糯米扫描我们 产品的二维码,选择商品,点击支付将会自动调用百度钱包进行支付,支付成功后返回成功页面,在我们的成功页面有关闭 网页按钮,点击关闭按钮或者返回按钮将会关闭页面. 在百度钱包中使用百度轻应用的BLightApp.closeWindow() 方法可以关闭,但是在百度糯米中,按道理可以使用呀, 但是根本就不行,改成传统的window.close()也不行.当时我们在网上搜索和百度糯米的官网中,都没找到解决方法.

新装linux系统/etc/sysconfig目录下无iptables文件的解决方法

今天新装了Linux,希望去做些防火墙的策略,使用service iptables status查看防火墙的状态时,无任何反应,且使用service iptables start也启动不聊.后来发现在/etc/sysconfig目录下没有iptables文件(防火墙的策略一般都写在此文件中). 原因:在新安装的Linux系统中,防火墙默认是被禁掉的,一般也没有配置过任何防火墙的策略,所有不存在/etc/sysconfig/iptables文件. 解决方法: 在控制台使用iptables命令随便写

IDEA Debug模式下改变各类型变量值的方法

本文介绍了IDEA Debug模式下改变各类型变量值的方法,分享给大家,具体如下: 测试类 import org.slf4j.LoggerFactory; import java.util.HashMap; import java.util.Map; /** * Created by PengHongfu 2018-04-18 18:21 */ public class testClass { private static final org.slf4j.Logger logger = Logg

php cli模式下获取参数的方法

PHP在cli模式下接收参数有两种方法 1.使用argv数组 2.使用getopt方法 1.使用argv数组 例如:需要执行一个php,并传递三个参数(type=news, is_hot=1, limit=5) 创建test.php <?php print_r($argv); ?> 在命令行执行 php test.php news 1 5 输出: Array ( [0] => test.php [1] => news [2] => 1 [3] => 5 ) 可以看到ar

简单讲解Android开发中触摸和点击事件的相关编程方法

在Android上,不止一个途径来侦听用户和应用程序之间交互的事件.对于用户界面里的事件,侦听方法就是从与用户交互的特定视图对象截获这些事件.视图类提供了相应的手段. 在各种用来组建布局的视图类里面,你可能会注意到一些公共的回调方法看起来对用户界面事件有用.这些方法在该对象的相关动作发生时被Android框架调用.比如,当一个视图(如一个按钮)被触摸时,该对象上的onTouchEvent()方法会被调用.不过,为了侦听这个事件,你必须扩展这个类并重写该方法.很明显,扩展每个你想使用的视图对象(只

Android系统音量条实例代码

最近在定制Android系统音量条,发现代码还是蛮多的,下面总结一下. 代码是基于5.1.1版本的. 系统音量条的代码是在/frameworks/base/packages/SystemUI/src/com/android/systemui/volume/VolumePanel.java 布局文件是在/frameworks/base/packages/SystemUI/res/layout下. 先看看原生的音量条样式: 在代码中可以发现volume_dialog.xml这个文件,这个文件就是承载

Android系统中的蓝牙连接程序编写实例教程

Bluetooth结构 1.JAVA层 frameworks/base/core/java/android/bluetooth/ 包含了bluetooth的JAVA类. 2.JNI层 frameworks/base/core/jni/android_bluetooth_开头的文件 定义了bluez通过JNI到上层的接口. frameworks/base/core/jni/android_server_bluetoothservice.cpp 调用硬件适配层的接口system/bluetooth/