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

时间: 2024-12-26 20:20:32

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

Android中的ViewPager视图滑动切换类的入门实例教程_Android

ViewPager引入示例首先让大家有个全局的认识,直接上个项目,看看仅仅通过这几行代码,竟然就能完成如此强悍的功能. 效果图: 实现了三个view间的相互滑动. 第一个VIEW向第二个VIEW滑动.第二个VIEW向第三个VIEW滑动 一.新建项目,引入ViewPager控件ViewPager.它是google SDk中自带的一个附加包的一个类,可以用来实现屏幕间的切换. 1.在主布局文件里加入 <RelativeLayout xmlns:android="http://schemas.a

Android中的ViewPager视图滑动切换类的入门实例教程

ViewPager引入示例 首先让大家有个全局的认识,直接上个项目,看看仅仅通过这几行代码,竟然就能完成如此强悍的功能. 效果图: 实现了三个view间的相互滑动. 第一个VIEW向第二个VIEW滑动.第二个VIEW向第三个VIEW滑动 一.新建项目,引入ViewPager控件 ViewPager.它是google SDk中自带的一个附加包的一个类,可以用来实现屏幕间的切换. 1.在主布局文件里加入 <RelativeLayout xmlns:android="http://schemas

java-请问:用Java代码中实现在一个类编写两种方法,下面的代码接下来怎么办呢?谢谢!

问题描述 请问:用Java代码中实现在一个类编写两种方法,下面的代码接下来怎么办呢?谢谢! 解决方案 啥东西,写两个方法,那你就写两个函数就行了,什么怎么办 解决方案二: 你的类里不就2个方法了吗

Android 中Popwindow弹出菜单的两种方法实例

Android 中Popwindow弹出菜单的两种方法实例 1.popWindow就是对话框的一种方式! 此文讲解的android中对话框的一种使用方式,它叫popWindow. 2.popWindow的特性 Android的对话框有两种:PopupWindow和AlertDialog.它们的不同点在于: AlertDialog的位置固定,而PopupWindow的位置可以随意. AlertDialog是非阻塞线程的,而PopupWindow是阻塞线程的. PopupWindow的位置按照有无偏

Android中系统自带锁WalkLock与KeyguardLock用法实例详解

本文实例讲述了Android中系统自带锁WalkLock与KeyguardLock用法.分享给大家供大家参考,具体如下: WalkLock - 顾名思义 唤醒锁 点亮屏幕用的 KeyguardLock - 顾名思义 键盘锁 解锁键盘用的 详细介绍: 1: WalkLock 唤醒锁 - WalkLock真的能点亮屏幕吗? 答案是肯定的. 可是有时候为什么不点亮屏幕,这个就是参数设置的问题了. 复制代码 代码如下:PowerManager.newWakeLock(PowerManager.FULL_

Android 中API之Drawable资源详解及简单实例

Android 中API之Drawable资源 1.最常用的StateListDrawable 说StateListDrawable,很多Android猿可能感到不太熟悉,不过如果说selector选择器,肯定都会恍然大悟,不错,这两个东西就是同一个~~ 它的用途之广,每个app必用,下面就写一个demo,来简要说一下用法. 比如一个登陆界面,它的输入框在获取焦点时需要更改背景,登陆按钮在输入框中有内容时,则更改背景颜色,这时候用selector选择器,那就方便多了,效果如下: EditText

Android中Spinner控件之键值对用法实例分析_Android

本文实例讲述了Android中Spinner控件之键值对用法.分享给大家供大家参考.具体如下: 一.字典表,用来存放键值对信息 package com.ljq.activity; import java.io.Serializable; @SuppressWarnings("serial") public class Dict implements Serializable { private Integer id; private String text; public Dict()

Android中Spinner控件之键值对用法实例分析

本文实例讲述了Android中Spinner控件之键值对用法.分享给大家供大家参考.具体如下: 一.字典表,用来存放键值对信息 package com.ljq.activity; import java.io.Serializable; @SuppressWarnings("serial") public class Dict implements Serializable { private Integer id; private String text; public Dict()

广发银行手机银行APP转账步骤教程

给各位广发银行手机银行APP软件的使用者们来详细的解析分享一下转账的步骤. 步骤分享: 不管您使用的是什么银行的手机银行,第一步都是在自己的Android/iOS手机里下载并安装一个手机银行客户端. 广发银行手机银行跟广发银行网上银行不是同一个账号,得重新注册一下,因此第一次使用广发银行手机银行的朋友要先注册一个账号,默认是手机号登录. 打开广发银行手机银行客户端就提示手机银行转账免手续费. 成功登录广发银行手机银行客户端后,点击"转账汇款",再输入对方的账号,就可以进行转账了.