概述:
  做过Android Camera图像采集和处理的朋友们应该都知道,Android手机相机采集的原始帧(RawFrame)默认是横屏格式的,而官方API有没有提供一个设置Camera采集图像的方向的方法,导致我们拿到原始帧后还需要再次对其进行转换为对应需求的数据,例如YUV的格式,图像的方向等(旋转多少度合适),下面我就粗略的介绍一下大致的流程,理解浅薄,大神请勿喷。
     注意:当前还都是基于API<21的内容,如果压根不用android.hardware.Camera的话可能有区别,还没研究过Camera2是什么原理,这里先不介绍了。
onPreviewFrame的原理:
     1. 原理及一般使用流程:
         大致流程:
         Camera — 取数据(onPreviewFrame(Byte[] rawFrameData, Camera camera)) —> 原始帧处理(Rotate/Scale:使用Libyuv/FFmpeg等工具库)—> 编码器编码得到相应的h24数据 —> 发送给流媒体服务器 
 
原始帧的采集:
     1. 方向问题:
         这里说明一下采集得到的帧的方向问题:Android默认是左横屏状态取景,也就意味着摄像头采集的数据默认就是横屏的,如图1,而且没有对应的方法来设置传感器的采集方向,所以想要实现竖直拍摄取景就得经过Rotate(旋转90°或者270°)得到图2那样的效果。

图 1
 

 
图 2
 
原始帧的分析:类型、YUV格式:
      之前写了一篇博客,关于YUV的区别的,感兴趣的朋友可以看看:http://www.cnblogs.com/raomengyang/p/4924787.html
 
处理原始帧:
  废话不多说了,上代码吧。
  MainActivity.java: 将Camera采集的图像显示到屏幕,并且图像经过90度的旋转成竖屏。
package com.dreamy.cameraframedemo;

import android.app.Activity;
import android.graphics.ImageFormat;
import android.hardware.Camera;
import android.os.Bundle;
import android.view.Surface;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View; import com.dreamy.cameraframedemo.util.ImageUtils; import java.io.IOException; public class MainActivity extends Activity implements SurfaceHolder.Callback,
Camera.PreviewCallback, View.OnClickListener {
// raw frame resolution: 1280x720, image format is: YV12
// you need get all resolution that supported on your devices;
// my phone is HUAWEI honor 6Plus, most devices can use 1280x720
private static final int SRC_FRAME_WIDTH = 1280;
private static final int SRC_FRAME_HEIGHT = 720;
private static final int IMAGE_FORMAT = ImageFormat.YV12; private Camera mCamera;
private Camera.Parameters mParams;
private SurfaceView mSurfaceView;
private SurfaceHolder mSurfaceHolder; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
setListener();
} private void initView() {
mSurfaceView = (SurfaceView) findViewById(R.id.sv_recording);
mSurfaceHolder = mSurfaceView.getHolder();
mSurfaceHolder.setFixedSize(SRC_FRAME_WIDTH, SRC_FRAME_HEIGHT);
mSurfaceHolder.addCallback(this);
mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
} private void setListener() {
// set Listener if you want, eg: onClickListener
} @Override
public void onClick(View v) {
} @Override
public void onPreviewFrame(byte[] data, Camera camera) {
ImageUtils.saveImageData(data);
camera.addCallbackBuffer(data);
} private void openCamera(SurfaceHolder holder) {
releaseCamera(); // release Camera, if not release camera before call camera, it will be locked
mCamera = Camera.open(Camera.CameraInfo.CAMERA_FACING_BACK);
mParams = mCamera.getParameters();
setCameraDisplayOrientation(this, Camera.CameraInfo.CAMERA_FACING_BACK, mCamera);
mParams.setPreviewSize(SRC_FRAME_WIDTH, SRC_FRAME_HEIGHT);
mParams.setPreviewFormat(IMAGE_FORMAT); // setting preview format:YV12
mParams.setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_VIDEO);
mCamera.setParameters(mParams); // setting camera parameters
try {
mCamera.setPreviewDisplay(holder);
} catch (IOException ioe) {
ioe.printStackTrace();
}
mCamera.setPreviewCallback(this);
mCamera.startPreview();
} private synchronized void releaseCamera() {
if (mCamera != null) {
try {
mCamera.setPreviewCallback(null);
} catch (Exception e) {
e.printStackTrace();
}
try {
mCamera.stopPreview();
} catch (Exception e) {
e.printStackTrace();
}
try {
mCamera.release();
} catch (Exception e) {
e.printStackTrace();
}
mCamera = null;
}
} /**
* Android API: Display Orientation Setting
* Just change screen display orientation,
* the rawFrame data never be changed.
*/
private void setCameraDisplayOrientation(Activity activity, int cameraId, Camera camera) {
Camera.CameraInfo info = new Camera.CameraInfo();
Camera.getCameraInfo(cameraId, info);
int rotation = activity.getWindowManager().getDefaultDisplay()
.getRotation();
int degrees = 0;
switch (rotation) {
case Surface.ROTATION_0:
degrees = 0;
break;
case Surface.ROTATION_90:
degrees = 90;
break;
case Surface.ROTATION_180:
degrees = 180;
break;
case Surface.ROTATION_270:
degrees = 270;
break;
}
int displayDegree;
if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
displayDegree = (info.orientation + degrees) % 360;
displayDegree = (360 - displayDegree) % 360; // compensate the mirror
} else {
displayDegree = (info.orientation - degrees + 360) % 360;
}
camera.setDisplayOrientation(displayDegree);
} @Override
public void surfaceCreated(SurfaceHolder holder) {
openCamera(holder); // open camera
} @Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
}
}
activity_main.xml:
<?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"> <SurfaceView
android:id="@+id/sv_recording"
android:layout_width="match_parent"
android:layout_height="match_parent" />
</RelativeLayout>

别忘了在AndroidManifest中打开Camera的权限:

<uses-permission android:name="android.permission.CAMERA" />
<uses-feature android:name="android.hardware.camera.autofocus" />
<uses-feature android:name="android.hardware.camera" />
保存原始帧为图片格式:
package com.dreamy.cameraframedemo.util;

import android.os.Environment;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.text.SimpleDateFormat;
import java.util.Date; /**
* Created by raomengyang on 4/25/16.
*/
public class ImageUtils {
public static final int MEDIA_TYPE_IMAGE = 1;
public static final int MEDIA_TYPE_VIDEO = 2; // save image to sdcard path: Pictures/MyTestImage/
public static void saveImageData(byte[] imageData) {
File imageFile = getOutputMediaFile(MEDIA_TYPE_IMAGE);
if (imageFile == null) return;
try {
FileOutputStream fos = new FileOutputStream(imageFile);
fos.write(imageData);
fos.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
} public static File getOutputMediaFile(int type) {
File imageFileDir =
new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_PICTURES), "MyTestImage");
if (!imageFileDir.exists()) {
if (!imageFileDir.mkdirs()) {
return null;
}
}
// Create a media file name
String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss").format(new Date());
File imageFile;
if (type == MEDIA_TYPE_IMAGE) {
imageFile = new File(imageFileDir.getPath() + File.separator +
"IMG_" + timeStamp + ".jpg");
} else if (type == MEDIA_TYPE_VIDEO) {
imageFile = new File(imageFileDir.getPath() + File.separator +
"VID_" + timeStamp + ".mp4");
} else return null;
return imageFile;
}
}   

通过javah -classpath ./ com.dreamy.jni.LibyuvUtils 生成jni的h头文件LibyuvUtils.h
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_dreamy_jni_LibyuvUtils */ #ifndef _Included_com_dreamy_jni_LibyuvUtils
#define _Included_com_dreamy_jni_LibyuvUtils
#ifdef __cplusplus
extern "C" {
#endif
/*
* Class: com_dreamy_jni_LibyuvUtils
* Method: initRotateNV21
* Signature: (II)V
*/
JNIEXPORT void JNICALL Java_com_dreamy_jni_LibyuvUtils_initRotateNV21
(JNIEnv *, jclass, jint, jint); /*
* Class: com_dreamy_jni_LibyuvUtils
* Method: ScaleYV12ToI420
* Signature: ([B[BIIIII)V
*/
JNIEXPORT void JNICALL Java_com_dreamy_jni_LibyuvUtils_ScaleYV12ToI420
(JNIEnv *, jclass, jbyteArray, jbyteArray, jint, jint, jint, jint, jint); /*
* Class: com_dreamy_jni_LibyuvUtils
* Method: ReleaseRotateNV21
* Signature: ()V
*/
JNIEXPORT void JNICALL Java_com_dreamy_jni_LibyuvUtils_ReleaseRotateNV21
(JNIEnv *, jclass); #ifdef __cplusplus
}
#endif
#endif 

资源下载:

  YUV格式查看工具:RawViewerhttp://download.csdn.net/detail/zxccxzzxz/9508288

     项目地址:https://github.com/eterrao/BlogExamples.git

   名称:CameraFrameDemo
 
  接下来会写Libyuv对原始帧进行Rotate和Scale的使用方法。

  

 

【Android】Android Camera原始帧格式转换 —— 获取Camera图像(一)的更多相关文章

  1. android 中如何获取camera当前状态

    /** * 测试当前摄像头能否被使用 * * @return */ public static boolean isCameraCanUse() { boolean canUse = true; Ca ...

  2. [Android]android.graphics.Camera实现图像的旋转、缩放,配合Matrix实现图像的倾斜

    android.graphics.Camera可以对图像执行一些比较复杂的操作,诸如旋转与绽放,与Matrix可实现图像的倾斜. 个人总结Camera与Matrix的一些区别如下: Camera的ro ...

  3. Android 音视频开发(四):使用 Camera API 采集视频数据

    本文主要将的是:使用 Camera API 采集视频数据并保存到文件,分别使用 SurfaceView.TextureView 来预览 Camera 数据,取到 NV21 的数据回调. 注: 需要权限 ...

  4. 【转】Android Camera(五)使用Camera功能 AREA的理解

    http://blog.csdn.net/think_soft/article/details/7998478 使用Camera功能 大多数的Camera功能都是使用Camera.Parameters ...

  5. Android tabhost下的activity怎样获取传来的值

    android tabhost下的activity怎样获取传来的值,具体解决方案如下: 解决方案: 其他activity设置intent:Intent intent=new Intent(); int ...

  6. Android 4.0以后正确的获取外部sd卡存储目录

    刚解决这个棘手的问题 找了很久,随笔记下. 网上搜索 android 获取外部sd卡存储目录 普遍都是: 1) Environment.getExternalStorageDirectory() 这个 ...

  7. Android如何在初始化的时候获取加载的布局的宽高

    在自定义ListView中,需要将下拉刷新的View在初始化的时候设置padding隐藏起来,这时就要在初始化的时候获得要加载的布局View的高度. private View headView; he ...

  8. Android中从SD卡中获取歌词并与歌曲同步

    先看看效果图吧,再看代码 转换文件的编码格式 package com.xm; import java.io.BufferedInputStream; import java.io.BufferedRe ...

  9. Android ADB工具-操作手机和获取手设备信息(四)

    Android ADB工具-操作手机和获取手设备信息(四) 标签(空格分隔): Android ADB 6. 其它命令 命令 功能 adb shell input text <content&g ...

随机推荐

  1. (转载)JAVA线程池管理

    平时的开发中线程是个少不了的东西,比如tomcat里的servlet就是线程,没有线程我们如何提供多用户访问呢?不过很多刚开始接触线程的开发攻城师却在这个上面吃了不少苦头.怎么做一套简便的线程开发模式 ...

  2. PAT/查找元素习题集

    B1004. 成绩排名 (20) Description: 读入n名学生的姓名.学号.成绩,分别输出成绩最高和成绩最低学生的姓名和学号. Input: 每个测试输入包含1个测试用例,格式为: 第1行: ...

  3. 备库Seconds_Behind_Master的计算

    背景 在mysql主备环境下,主备同步过程如下,主库更新产生binlog, 备库io线程拉取主库binlog生成relay log.备库sql线程执行relay log从而保持和主库同步. 理论上主库 ...

  4. vs的dll引用机制

    vs2012编译的时候,遇到一个问题就是项目A中运行时缺失dll的问题,项目A引用类库B,类库B引用了x,y等dll,编译A项目的时候,出现x没拷贝到bin 目录. 通过跟踪编译输出发现,x没拷贝的原 ...

  5. ASP.NET中一种超简单的Ajax解决方案

    为什么是Ajax2? 因为之前有一个blqw.Ajax,并且已经在项目中投入使用了,但是没有这个方便,这个是后来才弄的,为了纪念第一版的blqw.Ajax,所以这个就2了... 话说看了评论才发现,原 ...

  6. Lucene系列-分析器

    分析器介绍 搜索的基础是对文本信息进行分析,Lucene的分析工具在org.apache.lucene.analysis包中.分析器负责对文本进行分词.语言处理得到词条,建索引和搜索的时候都需要用到分 ...

  7. Amazon Dynamo论文学习

    Dynamo是一个key-value数据存储系统,去中心化.高可扩展.高可用,使用一致性哈希来分区和备份数据,使用数据版本化来实现一致性. 核心技术 CAP:一致性.可用性.扩展性 一致性哈希:切分数 ...

  8. LINQ to Entities不识别方法***,因此该方法无法转换为存储表达式

    我的程序里有这么一段代码: order.OrderExpressInfo = (from oei in orderExpressRepository.Entities where oei.OrderI ...

  9. GUID相关知识

      全局唯一标识符(GUID,Globally Unique Identifier)是一种由算法生成的二进制长度为128位的数字标识符.GUID主要用于在拥有多个节点.多台计算机的网络或系统中.在理想 ...

  10. 大数据时代的IT架构设计

    大数据时代的IT架构设计(来自互联网.银行等领域的一线架构师先进经验分享) IT架构设计研究组 编著   ISBN 978-7-121-22605-2 2014年4月出版 定价:49.00元 208页 ...