在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. C# 多线程网络爬虫

    原文 C#制作多线程处理强化版网络爬虫 上次做了一个帮公司妹子做了爬虫,不是很精致,这次公司项目里要用到,于是有做了一番修改,功能添加了网址图片采集,下载,线程处理界面网址图片下载等. 说说思路:首相 ...

  2. java web 学习二(Tomcat服务器学习和使用1)

    一.Tomcat服务器端口的配置 Tomcat的所有配置都放在conf文件夹之中,里面的server.xml文件是配置的核心文件. 如果想修改Tomcat服务器的启动端口,则可以在server.xml ...

  3. <转>Python运行的17个时新手常见错误小结

    1)忘记在 if , elif , else , for , while , class ,def 声明末尾添加 :(导致 “SyntaxError :invalid syntax”) 该错误将发生在 ...

  4. PHP 调用外部程序的几种方式

    /* php 调用python 的代码 // 第一种: // echo passthru('C:/Python34/PY.exe D:/do.py'); // 第二种: // echo exec('C ...

  5. HDU5634 Rikka with Phi 线段树

    // HDU5634 Rikka with Phi 线段树 // 思路:操作1的时候,判断一下当前区间是不是每个数都相等,在每个数相等的区间上操作.相当于lazy,不必更新到底. #include & ...

  6. poj1000 A+B Problem

    Description Calculate a+b Input Two integer a,b (0<=a,b<=10) Output Output a+b Sample Input 1 ...

  7. Hadoop学习之--Capaycity Scheduler源码分析

    Capacity Scheduler调度策略当一个新的job是否允许添加到队列中进行初始化,判断当前队列和用户是否已经达到了初始化数目的上限,下面就从代码层面详细介绍整个的判断逻辑.Capaycity ...

  8. Unity中的单实例

    static GUIManager myInstance; public static GUIManager Instance { get { if (myInstance == null) myIn ...

  9. es基础操作

    在curl 的 url 中 , 问号后台可以加上pretty=true , 可以将返回来的json进行格式化 . 如果es集群中只有一个node , 那么他的集群健康状态是黄色的 , 只需要再加一个n ...

  10. matlab search path

    What Is the MATLAB Search PathThe search path, or path is a subset of all the folders in the file sy ...