在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,并在里面实现了照相机的预览功能.这个我觉得最简单的照相机预览代码:

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 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;
Log.d("ddd", "4");
}
}

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.

@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()来重新开启预览。

  我用以上知识,加到上面的那个例子,就形成了下面的代码:

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

 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;并把一些没有用到的代码全部删掉,剩下一些必须的代码

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

Android 浅谈相机研发的更多相关文章

  1. Android 浅谈 设计与屏幕适配 【1.6235449734285716】

    extends: http://www.ui.cn/detail/45435.html http://www.2cto.com/kf/201501/372699.html http://www.cnb ...

  2. android 浅谈Aidl 通讯机制

    服务端: 首先是编写一个aidl文件,注意AIDL只支持方法,不能定义静态成员,并且方法也不能有类似public等的修饰符:AIDL运行方法有任何类型的参数和返回值,在java的类型中,以下的类型使用 ...

  3. [iOS、Unity、Android] 浅谈闭包的使用方法

    前言 我们经常所编程语言的的进步速度是落后于硬件的发展速度的. 但是最近几年,闭包语法在各个语言中都有自己的体现形式,例如 • C语言中使用函数指针作为回调函数的入口: • Java和C#语言中的La ...

  4. Android——浅谈HTTP中Get与Post的区别(转)

    原文地址:http://network.51cto.com/art/201407/446434.htm Http定义了与服务器交互的不同方法,最基本的方法有4种,分别是GET,POST,PUT,DEL ...

  5. 浅谈Android应用性能之内存

    本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 文/ jaunty [博主导读]在Android开发中,不免会遇到许多OOM现象,一方面可能是由于开 ...

  6. 浅谈Android编码规范及命名规范

    前言: 目前工作负责两个医疗APP项目的开发,同时使用LeanCloud进行云端配合开发,完全单挑. 现大框架已经完成,正在进行细节模块上的开发 抽空总结一下Android项目的开发规范:1.编码规范 ...

  7. Android安全开发之浅谈密钥硬编码

    Android安全开发之浅谈密钥硬编码 作者:伊樵.呆狐@阿里聚安全 1 简介 在阿里聚安全的漏洞扫描器中和人工APP安全审计中,经常发现有开发者将密钥硬编码在Java代码.文件中,这样做会引起很大风 ...

  8. 浅谈Android应用保护(一):Android应用逆向的基本方法

    对于未进行保护的Android应用,有很多方法和思路对其进行逆向分析和攻击.使用一些基本的方法,就可以打破对应用安全非常重要的机密性和完整性,实现获取其内部代码.数据,修改其代码逻辑和机制等操作.这篇 ...

  9. Android应用安全开发之浅谈加密算法的坑

      <Android应用安全开发之浅谈加密算法的坑> 作者:阿里移动安全@伊樵,@舟海 阿里聚安全,一站式解决应用开发安全问题     Android开发中,难免会遇到需要加解密一些数据内 ...

随机推荐

  1. ISAPI在IIS7上的配置

    主要介绍ISAPI的作用.ISAPI在IIS7上的配置.开发ISAPI的基本内容及使用VS 2008配置ISAPI DLL开发项目. 一.ISAPI介绍 缩写词=Internet Server App ...

  2. codeforces 691D Swaps in Permutation DFS

    这个题刚开始我以为是每个交换只能用一次,然后一共m次操作 结果这个题的意思是操作数目不限,每个交换也可以无限次 所以可以交换的两个位置连边,只要两个位置连通,就可以呼唤 然后连通块内排序就好了 #in ...

  3. delegate 为什么用 weak属性

    weak指针主要用于“父-子”关系,父亲拥有一个儿子的strong指针,因此是儿子的所有者:但是为了阻止所有权回环,儿子需要使用weak指针指向父亲:你的viewcontroller通过strong指 ...

  4. 百家搜索:在网站中添加Google、百度等搜索引擎

    来源:http://www.ido321.com/1143.html 看到一些网站上添加了各种搜索引擎.如Google.百度.360.有道等,就有点好奇,这个怎么实现?研究了一各个搜索引擎怎么传送关键 ...

  5. php里面为什么header之前有输出报错 源码分析

    众所周知,php 里面 header之前有输出的话,会报错,例如下面这样   就这个错误,我们开始查阅php源代码,到底是怎样做的,至于php源代码分析,安装,和调试时怎样配置的,我会专门写一篇文章去 ...

  6. java动态代理与老式AOP实现

    JAVA的动态代理 代理模式是常用的java设计模式,他的特征是代理类与委托类有同样的接口,代理类主要负责为委托类预处理消息.过滤消息.把消息转发给委托类,以及事后处理消息等.代理类与委托类之间通常会 ...

  7. matlab中的字符串数组与函数调用

    1, matlab中的字符串就是1维字符数组,即如: a = 'dddssd'; b = 'lsde'; c = [a, b]; 当然也可以: c= strcat(a, b); 2, matlab中的 ...

  8. Glibc辅助运行库 (C RunTime Library): crt0.o,crt1.o,crti.o crtn.o,crtbegin.o crtend.o

    crt1.o, crti.o, crtbegin.o, crtend.o, crtn.o 等目标文件和daemon.o(由我们自己的C程序文件产生)链接成一个执行文件.前面这5个目标文件的作用分别是启 ...

  9. 第二百七十四、五、六天 how can I 坚持

    三天小长假这么快就过去了,好快啊.基本都是在济南过的. 元旦.坐车回济南.下午在万科新里程看了一下午房子,没有买啊,93的现在八千六七,有点贵啊,户型也不是自己喜欢的. 晚上一块吃了个饭,还行,晚上在 ...

  10. iOS事件机制(一)

    运用的前提是掌握 掌握的本质是理解 本篇内容将围绕iOS中事件及其传递机制进行学习和分析.在iOS中,事件分为三类: 触控事件(单点.多点触控以及各种手势操作) 传感器事件(重力.加速度传感器等) 远 ...