android之相机开发
android之相机开发
在android中应用相机功能,一般有两种:一种是直接调用系统相机,一种自己写的相机。
我将分别演示两种方式的使用:
第一种:是使用Intent跳转到系统相机,action为:android.media.action.STILL_IMAGE_CAMERA
关键代码:
- Intent intent = new Intent(); //调用照相机
- intent.setAction("android.media.action.STILL_IMAGE_CAMERA");
- startActivity(intent);
例子:
- import android.app.Activity;
- import android.content.Intent;
- import android.os.Bundle;
- public class CameraTest_2 extends Activity {
- /** Called when the activity is first created. */
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- Intent intent = new Intent(); //调用照相机
- intent.setAction("android.media.action.STILL_IMAGE_CAMERA");
- startActivity(intent);
- }
- }
想要测试的,可以直接新建一个项目,并且把主activity的代码换成上面的,然后运行,我测试了一下,上面这个代码并不
需要权限,毕竟只是调用系统自带的程序。
当然网上还有一些其他相关的调用方法,只要设置对了action,那么系统就会调用系统自带的相机.
第二种:
(1)首先我们要自己创建一个照相界面,必须考虑用什么控件显示照相机中的预览效果,显然android已经帮我们做好了选择,那就是SurfaceView
控制SurfaceView需要一个surfaceHolder,他是系统提供的一个用来设置SurfaceView的一个对象,可以通过surfaceView.getHolder()这个方法来获得.
Camera提供一个setPreviewDisplay(SurfaceHolder)的方法来连接surfaceHolder,并通过他来控制surfaceView.
而我们则使用android的Camera类提供了startPreview()和stopPreview()来开启和关闭预览.
关系如下:
Camera -- -->SurfaceHolder------>SurfaceView.
(2)知道怎么预览了,当然也要知道怎么开启相机.Camera.open()这是个静态方法,如果相机没有别人用着,则会返回一个 相机引用,如果被人用着,则会抛出异常。很奇怪的是,这个方法,不能随便放,如放在构造方法或者onCreate()方法中,都会照成没有预览效果.
(3)
SurfaceHolder.Callback,这是个holder用来显示surfaceView 数据的接口,他分别必须实现3个方法
surfaceCreated()这个方法是surface 被创建后调用的
surfaceChanged()这个方法是当surfaceView发生改变后调用的
surfaceDestroyed()这个是当surfaceView销毁时调用的.
surfaceHolde通过addCallBack()方法将响应的接口绑定到他身上.
surfaceHolder还必须设定一个setType()方法,查看api的时候,发现这个方法已经过时,但是不写,又会报错
(4)
我用以上知识写了一个MySurfaceView类,他继承于SurfaceView,并在里面实现了照相机的预览功能.这个我觉得最简单的照相机预览代码:
MySurfaceView.java:
- import java.io.IOException;
- import android.content.Context;
- import android.graphics.PixelFormat;
- import android.hardware.Camera;
- import android.util.Log;
- import android.view.SurfaceHolder;
- import android.view.SurfaceView;
- public class MySurfaceView extends SurfaceView implements SurfaceHolder.Callback{
- SurfaceHolder holder;
- Camera myCamera;
- public MySurfaceView(Context context)
- {
- super(context);
- holder = getHolder();//获得surfaceHolder引用
- holder.addCallback(this);
- holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);//设置类型
- }
- @Override
- public void surfaceCreated(SurfaceHolder holder) {
- // TODO Auto-generated method stub
- if(myCamera == null)
- {
- myCamera = Camera.open();//开启相机,不能放在构造函数中,不然不会显示画面.
- try {
- myCamera.setPreviewDisplay(holder);
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- @Override
- public void surfaceChanged(SurfaceHolder holder, int format, int width,
- int height) {
- myCamera.startPreview();
- }
- @Override
- public void surfaceDestroyed(SurfaceHolder holder) {
- // TODO Auto-generated method stub
- myCamera.stopPreview();//停止预览
- myCamera.release();//释放相机资源
- myCamera = null;
- Log.d("ddd", "4");
- }
- }
主activity代码:
- import android.app.Activity;
- import android.os.Bundle;
- import android.view.View;
- import android.view.View.OnClickListener;
- public class CameraTest_3 extends Activity {
- /** Called when the activity is first created. */
- MySurfaceView mySurface;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- mySurface = new MySurfaceView(this);
- setContentView(mySurface);
- }
- }
而且必须给应用添加权限:
- <uses-permission android:name="android.permission.CAMERA"></uses-permission>
(5)能够预览了,接下来就是拍照了,拍照用到了一个camera.tackPiture()这个方法,这个方法,有三个参数分别是
ShutterCallBack shutter,PictureCallBack raw,PictureCallBack jpeg.
下面是对他们的实现
- private ShutterCallback shutter = new ShutterCallback() {
- @Override
- public void onShutter() {
- // TODO Auto-generated method stub
- Log.d("ddd", "shutter");
- }
- };
- private PictureCallback raw = new PictureCallback() {
- @Override
- public void onPictureTaken(byte[] data, Camera camera) {
- // TODO Auto-generated method stub
- Log.d("ddd", "raw");
- }
- };
- private PictureCallback jpeg = new PictureCallback() {
- @Override
- public void onPictureTaken(byte[] data, Camera camera) {
- // TODO Auto-generated method stub
- Log.d("ddd","jpeg");
- }
- };
当开始拍照时,会依次调用shutter的onShutter()方法,raw的onPictureTaken方法,jpeg的onPictureTaken方法.
三个参数的作用是shutter--拍照瞬间调用,raw--获得没有压缩过的图片数据,jpeg---返回jpeg的图片数据
当你不需要对照片进行处理,可以直接用null代替.
注意,当调用camera.takePiture方法后,camera关闭了预览,这时需要调用startPreview()来重新开启预览。
我用以上知识,加到上面的那个例子,就形成了下面的代码:
MySurfaceView.java:
- package jason.camera;
- import java.io.IOException;
- import android.content.Context;
- import android.graphics.PixelFormat;
- import android.hardware.Camera;
- import android.hardware.Camera.PictureCallback;
- import android.hardware.Camera.ShutterCallback;
- import android.util.Log;
- import android.view.SurfaceHolder;
- import android.view.SurfaceView;
- public class MySurfaceView extends SurfaceView implements SurfaceHolder.Callback{
- SurfaceHolder holder;
- Camera myCamera;
- private ShutterCallback shutter = new ShutterCallback() {
- @Override
- public void onShutter() {
- // TODO Auto-generated method stub
- Log.d("ddd", "shutter");
- }
- };
- private PictureCallback raw = new PictureCallback() {
- @Override
- public void onPictureTaken(byte[] data, Camera camera) {
- // TODO Auto-generated method stub
- Log.d("ddd", "raw");
- }
- };
- private PictureCallback jpeg = new PictureCallback() {
- @Override
- public void onPictureTaken(byte[] data, Camera camera) {
- // TODO Auto-generated method stub
- Log.d("ddd","jpeg");
- }
- };
- public MySurfaceView(Context context)
- {
- super(context);
- holder = getHolder();//获得surfaceHolder引用
- holder.addCallback(this);
- holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);//设置类型
- }
- public void tackPicture()
- {
- myCamera.takePicture(null,null,null);
- }
- public void voerTack()
- {
- myCamera.startPreview();
- }
- @Override
- public void surfaceChanged(SurfaceHolder holder, int format, int width,
- int height) {
- myCamera.startPreview();
- }
- @Override
- public void surfaceCreated(SurfaceHolder holder) {
- // TODO Auto-generated method stub
- if(myCamera == null)
- {
- myCamera = Camera.open();//开启相机,不能放在构造函数中,不然不会显示画面.
- try {
- myCamera.setPreviewDisplay(holder);
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- @Override
- public void surfaceDestroyed(SurfaceHolder holder) {
- // TODO Auto-generated method stub
- myCamera.stopPreview();//停止预览
- myCamera.release();//释放相机资源
- myCamera = null;
- }
- }
主activity:
- import android.app.Activity;
- import android.os.Bundle;
- import android.view.View;
- import android.view.View.OnClickListener;
- public class CameraTest_3 extends Activity implements OnClickListener {
- /** Called when the activity is first created. */
- MySurfaceView mySurface;
- boolean isClicked = false;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- mySurface = new MySurfaceView(this);
- setContentView(mySurface);
- mySurface.setOnClickListener(this);
- }
- @Override
- public void onClick(View v) {
- // TODO Auto-generated method stub
- if(!isClicked)
- {
- mySurface.tackPicture();
- isClicked = true;
- }else
- {
- mySurface.voerTack();
- isClicked = false;
- }
- }
这样就是实现了拍照的功能,那么怎样要图片保存呢?那么这是就需要在那个参数中的jpeg的
方法里面进行处理了,那个方法的data参数,就是相片的数据。
我们通过BitmapFactory.decodeByteArray(data, 0, data.length)来获得图片并通过io处理,将图片保存到想要保存的位置
下面这段代码,是将照片保存到/sdcard/wjh.jpg;并把一些没有用到的代码全部删掉,剩下一些必须的代码
- package java.camera;
- import java.io.BufferedInputStream;
- import java.io.BufferedOutputStream;
- import java.io.File;
- import java.io.FileOutputStream;
- import java.io.IOException;
- import java.io.OutputStream;
- import android.content.Context;
- import android.graphics.Bitmap;
- import android.graphics.BitmapFactory;
- import android.graphics.PixelFormat;
- import android.hardware.Camera;
- import android.hardware.Camera.PictureCallback;
- import android.hardware.Camera.ShutterCallback;
- import android.util.Log;
- import android.view.SurfaceHolder;
- import android.view.SurfaceView;
- public class MySurfaceView extends SurfaceView implements SurfaceHolder.Callback{
- SurfaceHolder holder;
- Camera myCamera;
- private PictureCallback jpeg = new PictureCallback() {
- @Override
- public void onPictureTaken(byte[] data, Camera camera) {
- // TODO Auto-generated method stub
- try
- {
- Bitmap bm = BitmapFactory.decodeByteArray(data, 0, data.length);
- File file = new File("/sdcard/wjh.jpg");
- BufferedOutputStream bos
- = new BufferedOutputStream(new FileOutputStream(file));
- bm.compress(Bitmap.CompressFormat.JPEG,100,bos);
- bos.flush();
- bos.close();
- }catch(Exception e)
- {
- e.printStackTrace();
- }
- }
- };
- public MySurfaceView(Context context)
- {
- super(context);
- holder = getHolder();//获得surfaceHolder引用
- holder.addCallback(this);
- holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);//设置类型
- }
- public void tackPicture()
- {
- myCamera.takePicture(null,null,jpeg);
- }
- public void voerTack()
- {
- myCamera.startPreview();
- }
- @Override
- public void surfaceChanged(SurfaceHolder holder, int format, int width,
- int height) {
- myCamera.startPreview();
- }
- @Override
- public void surfaceCreated(SurfaceHolder holder) {
- // TODO Auto-generated method stub
- if(myCamera == null)
- {
- myCamera = Camera.open();//开启相机,不能放在构造函数中,不然不会显示画面.
- try {
- myCamera.setPreviewDisplay(holder);
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- @Override
- public void surfaceDestroyed(SurfaceHolder holder) {
- // TODO Auto-generated method stub
- myCamera.stopPreview();//停止预览
- myCamera.release();//释放相机资源
- myCamera = null;
- }
- }
注意,这是必须添加在sd卡上写数据的权限
- <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
(7)能够拍照了,这下子要考虑如何让图片更好看了,这显然是专业人士的强项,但是我们在程序上,也可以做一些处理,
向上面的那些,因为我直接把surfaceView当做整体布局,就可能出现屏幕被拉开了,不是很好看,所以这时,就可以不要把
surfaceView弄成整体布局,把他弄到一个布局管理器,再设置相关的参数.
这时需要注意的是有些参数不能随便乱设
如以下代码:
- Camera.Parameters parames = myCamera.getParameters();//获得参数对象
- parames.setPictureFormat(PixelFormat.JPEG);//设置图片格式
- parames.setPreviewSize(640,480);//这里面的参数只能是几个特定的参数,否则会报错.(176*144,320*240,352*288,480*360,640*480)
- myCamera.setParameters(parames);
还有自动对焦,当然有些手机没有这个功能,自动对焦是通过autoFocus()这个方法调用一个自动对焦的接口,并在里面进行处理。
注意,这个方法必须在startPreview()和stopPreview()中间。
AutoFocusCallback是自动对焦的接口,实现它必须实现public void onAutoFocus(boolean success, Camera camera)这个方法,
所以我们可以将拍照方法放在这里面,然后对焦后再进行拍摄。。效果会好很多。
注意自动对焦需要添加
- <uses-feature android:name="android.hardware.camera.autofocus" />
下面是主要代码,直接贴出来了
- import java.io.BufferedOutputStream;
- import java.io.File;
- import java.io.FileOutputStream;
- import java.io.IOException;
- import android.app.Activity;
- import android.content.pm.ActivityInfo;
- import android.graphics.Bitmap;
- import android.graphics.BitmapFactory;
- import android.graphics.PixelFormat;
- import android.hardware.Camera;
- import android.hardware.Camera.AutoFocusCallback;
- import android.hardware.Camera.PictureCallback;
- import android.os.Bundle;
- import android.view.SurfaceHolder;
- import android.view.SurfaceView;
- import android.view.View;
- import android.view.Window;
- import android.view.SurfaceHolder.Callback;
- import android.view.View.OnClickListener;
- public class CameraTest_4 extends Activity implements
- Callback, OnClickListener, AutoFocusCallback{
- SurfaceView mySurfaceView;//surfaceView声明
- SurfaceHolder holder;//surfaceHolder声明
- Camera myCamera;//相机声明
- String filePath="/sdcard/wjh.jpg";//照片保存路径
- boolean isClicked = false;//是否点击标识
- //创建jpeg图片回调数据对象
- PictureCallback jpeg = new PictureCallback() {
- @Override
- public void onPictureTaken(byte[] data, Camera camera) {
- // TODO Auto-generated method stub
- try
- {// 获得图片
- Bitmap bm = BitmapFactory.decodeByteArray(data, 0, data.length);
- File file = new File(filePath);
- BufferedOutputStream bos =
- new BufferedOutputStream(new FileOutputStream(file));
- bm.compress(Bitmap.CompressFormat.JPEG, 100, bos);//将图片压缩到流中
- bos.flush();//输出
- bos.close();//关闭
- }catch(Exception e)
- {
- e.printStackTrace();
- }
- }
- };
- /** Called when the activity is first created. */
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- requestWindowFeature(Window.FEATURE_NO_TITLE);//无标题
- //设置拍摄方向
- this.setRequestedOrientation(ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE);
- setContentView(R.layout.main);
- //获得控件
- mySurfaceView = (SurfaceView)findViewById(R.id.surfaceView1);
- //获得句柄
- holder = mySurfaceView.getHolder();
- //添加回调
- holder.addCallback(this);
- //设置类型
- holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
- //设置监听
- mySurfaceView.setOnClickListener(this);
- }
- @Override
- public void surfaceChanged(SurfaceHolder holder, int format, int width,
- int height) {
- // TODO Auto-generated method stub
- //设置参数并开始预览
- Camera.Parameters params = myCamera.getParameters();
- params.setPictureFormat(PixelFormat.JPEG);
- params.setPreviewSize(640,480);
- myCamera.setParameters(params);
- myCamera.startPreview();
- }
- @Override
- public void surfaceCreated(SurfaceHolder holder) {
- // TODO Auto-generated method stub
- //开启相机
- if(myCamera == null)
- {
- myCamera = Camera.open();
- try {
- myCamera.setPreviewDisplay(holder);
- } catch (IOException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- @Override
- public void surfaceDestroyed(SurfaceHolder holder) {
- // TODO Auto-generated method stub
- //关闭预览并释放资源
- myCamera.stopPreview();
- myCamera.release();
- myCamera = null;
- }
- @Override
- public void onClick(View v) {
- // TODO Auto-generated method stub
- if(!isClicked)
- {
- myCamera.autoFocus(this);//自动对焦
- isClicked = true;
- }else
- {
- myCamera.startPreview();//开启预览
- isClicked = false;
- }
- }
- @Override
- public void onAutoFocus(boolean success, Camera camera) {
- // TODO Auto-generated method stub
- if(success)
- {
- //设置参数,并拍照
- Camera.Parameters params = myCamera.getParameters();
- params.setPictureFormat(PixelFormat.JPEG);
- params.setPreviewSize(640,480);
- myCamera.setParameters(params);
- myCamera.takePicture(null, null, jpeg);
- }
- }
- }
main.xml:
- <linearlayout xmlns:android="http://schemas.android.com/apk/res/android"
- androidrientation="vertical"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- >
- <surfaceview android:id="@+id/surfaceView1"
- android:layout_width="640px"
- android:layout_height="480px"
- android:layout_gravity="center">
AndroidManifest.xml:
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.wjh.camera"
- android:versionCode="1"
- android:versionName="1.0">
- <uses-sdk android:minSdkVersion="7" />
- <uses-permission android:name="android.permission.CAMERA"></uses-permission>
- <uses-feature android:name="android.hardware.camera.autofocus" />
- <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
- <application android:icon="@drawable/icon" android:label="@string/app_name">
- <activity android:name=".CameraTest_4"
- android:label="@string/app_name">
- <intent-filter>
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- </application>
- </manifest>
PS:其实自己去实现相机的必要性不大,大部分应用都是直接调用系统的相机,毕竟相机是涉及到一些比较专业的成像知识.
android之相机开发的更多相关文章
- 【转】Android Camera 相机开发详解
在Android 5.0(SDK 21)中,Google使用Camera2替代了Camera接口.Camera2在接口和架构上做了巨大的变动, 但是基于众所周知的原因,我们还必须基于 Android ...
- Android相机开发那些坑
版权声明:本文由王梓原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/168 来源:腾云阁 https://www.qclou ...
- android 自定义相机
老规矩,先上一下项目地址:GitHub:https://github.com/xiangzhihong/CameraDemo 方式: 调用Camera API 自定义相机 调用系统相机 由于需求不同, ...
- Android 音视频开发学习思路
Android 音视频开发这块目前的确没有比较系统的教程或者书籍,网上的博客文章也都是比较零散的.只能通过一点点的学习和积累把这块的知识串联积累起来. 初级入门篇: Android 音视频开发(一) ...
- Android音视频开发(1):H264 基本原理
前言 H264 视频压缩算法现在无疑是所有视频压缩技术中使用最广泛,最流行的.随着 x264/openh264 以及 ffmpeg 等开源库的推出,大多数使用者无需再对H264的细节做过多的研究,这大 ...
- Android应用安全开发之浅谈加密算法的坑
<Android应用安全开发之浅谈加密算法的坑> 作者:阿里移动安全@伊樵,@舟海 阿里聚安全,一站式解决应用开发安全问题 Android开发中,难免会遇到需要加解密一些数据内 ...
- [转]Android样式的开发:shape篇
转载自Keegan小钢原文链接:http://keeganlee.me/post/android/20150830 Android样式的开发:shape篇Android样式的开发:selector篇A ...
- Android之NDK开发(转)
Android之NDK开发 一.NDK产生的背景 Android平台从诞生起,就已经支持C.C++开发.众所周知,Android的SDK基于Java实现,这意味着基于Android SDK进行开发的第 ...
- Android移动APP开发笔记——最新版Cordova 5.3.1(PhoneGap)搭建开发环境
引言 简单介绍一下Cordova的来历,Cordova的前身叫PhoneGap,自被Adobe收购后交由Apache管理,并将其核心功能开源改名为Cordova.它能让你使用HTML5轻松调用本地AP ...
随机推荐
- aforge之hough
/// <summary> /// 参数不能是一位索引 /// </summary> /// <param name="bmp"></pa ...
- ThinkPHP模板
[MVC模式] M:Model 数据模型层,负责数据操作 V:View 视图层,负责显示视图 C:Controller 控制器,实现业务逻辑 tp框架url地址可以由以下四种 http://网址/in ...
- PowerDesigner 的mysql PDM 的COMMENT注释
PowerDesigner 的mysql PDM 的COMMENT注释 2012-11-01 15:38 4447人阅读 评论(0) 举报 分类: 数据库相关(7) PowerDesigner 的my ...
- url 取出文件扩展名
/**url 取出文件扩展名 *///方法一function getExt1($url) { $arr = parse_url($url); $file = basename($arr[' ...
- 恢复SQLSERVER被误删除的数据(转)
恢复SQLSERVER被误删除的数据 曾经想实现Log Explorer for SQL Server的功能,利用ldf里面的日志来还原误删除的数据 这里有一篇文章做到了,不过似乎不是所有的数据类型都 ...
- 10682 deathgod想知道的事(数论)
10682 deathgod想知道的事 该题有题解 时间限制:1000MS 内存限制:65535K提交次数:265 通过次数:14 题型: 编程题 语言: G++;GCC Description ...
- Heap Operations(模拟题)
Heap Operations time limit per test 1 second memory limit per test 256 megabytes input standard inp ...
- Emacs阅读chm文档
.title { text-align: center; margin-bottom: .2em } .subtitle { text-align: center; font-size: medium ...
- 关于LR监视Windows和linux的说明
一.监控windows系统: 1.监视连接前的准备工作 1)进入被监视windows系统,开启以下二个服务Remote Procedure Call(RPC) 和Remote Registry Ser ...
- C语言strtok()函数:字符串分割
头文件:#include <string.h> 定义函数:char * strtok(char *s, const char *delim); 函数说明:strtok()用来将字符串分割成 ...