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;
  } 

}
(0)

相关推荐

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

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

  • Android Camera变焦编程步骤

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

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

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

  • 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编程中调用Camera时预览画面有旋转问题的解决方法

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

  • 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实现Camera2预览和拍照效果

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

  • 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

随机推荐