在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. 2014年acm亚洲区域赛·鞍山站

    今天北京赛站的比赛也结束了···看了一天的直播之后意识到鞍山站的比赛都过去了一个多月了···这一个月比较萎靡···整天都在睡觉写报告画工图中度过··· 鞍山比哈尔滨还是暖和很多的···就是山上有奇怪的 ...

  2. [转] WinForm自定义函数FindControl实现按名称查找控件

    原文地址 WinForm自定义函数FindControl实现按名称查找控件 本文所述实例实现WinForm自定义函数FindControl实现按名称查找控件的功能,在C#程序开发中有一定的实用价值. ...

  3. C++重要知识点小结---1

    1.C++中类与结构的唯一区别是:类(class)定义中默认情况下的成员是private的,而结构(struct)定义中默认情况下的成员是public的. 2. ::叫作用域区分符,指明一个函数属于哪 ...

  4. slidingmenu + fragment 左右菜单滑动

                           content_frame.xml <?xml version="1.0" encoding="utf-8" ...

  5. SSH 连接慢的解决方案详解

    SSH 连接慢的解决方案详解 http://www.codeceo.com/article/ssh-slow.html

  6. 使用 gradle 编译多版本 android 应用

    最近要做一个 android 产品的变种版本,需要编出不同版本,每个版本有不同的包名.图标等等,和一些特有的逻辑. 很久之前做过类似的工作,当时没有 gradle, 用的方法是把公共代码抽成一个 li ...

  7. 关于Cocoapods安装与问题

    安装: 1.打开终端 2.如果网络没有FQ的话,需要通过淘宝的RubyGems镜像进行安装. 首先移除默认地址: gem sources --remove https://rubygems.org/ ...

  8. Asp.net MVC 处理文件的上传下载

    如果你仅仅只有Asp.net Web Forms背景转而学习Asp.net MVC的,我想你的第一个经历或许是那些曾经让你的编程变得愉悦无比的服务端控件都驾鹤西去了.FileUpload就是其中一个, ...

  9. 一起刷LeetCode2-Add Two Numbers

    今天看不进去论文,也学不进去新技术,于是先把题刷了,一会补别的. -----------------------------------------------------我才不是分割线------- ...

  10. 解决A program file was not specified in the launch configuration.问题

        问题描述: 在eclipse 中开发c++或c是比较麻烦的事情,刚刚配置好mingw32和cdt和环境变量后,新建一个hello world的c++项目还是会出现问题.主要是在编译的时候会提示 ...