使用 Android Camera API 完成音视频的采集、编码、封包成 mp4 输出
基于android.hardware.Camera,创建一个横屏应用,实时预览摄像头图像,实现录像并输出MP4的功能。

1、申请权限

<!-- 需要录制音视频权限和写外部存储权限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<uses-permission android:name="android.permission.CAMERA" />
<uses-feature android:name="android.hardware.camera" />

在activity中动态申请权限

private static final String[] VIDEO_PERMISSIONS = {
Manifest.permission.CAMERA,
Manifest.permission.RECORD_AUDIO,
Manifest.permission.WRITE_EXTERNAL_STORAGE
};

2、实现摄像头预览功能

使用SurfaceView来预览。新建CameraPreview类继承自SurfaceView并实现SurfaceHolder.Callback;Camera相关操作都放在这个View里。surfaceCreated中获取Camera实例,启动预览;设置预览相关参数,surfaceDestroyed释放Camera

public class CameraPreview extends SurfaceView implements SurfaceHolder.Callback {
private SurfaceHolder mHolder;
private Camera mCamera;
public static final int MEDIA_TYPE_IMAGE = 1;
public static final int MEDIA_TYPE_VIDEO = 2;
private static int mOptVideoWidth = 1920; // 默认视频帧宽度
private static int mOptVideoHeight = 1080;
private Uri outputMediaFileUri;
private String outputMediaFileType;
public CameraPreview(Context context) {
super(context);
mHolder = getHolder();
mHolder.addCallback(this);
}
private static Camera getCameraInstance() {
Camera c = null;
try {
c = Camera.open();
} catch (Exception e) {
Log.d(TAG, "camera is not available");
}
return c;
}@Override
public void surfaceCreated(SurfaceHolder holder) {
mCamera = getCameraInstance();
try {
mCamera.setPreviewDisplay(holder);
mCamera.startPreview();
getCameraOptimalVideoSize(); // 找到最合适的分辨率
} catch (IOException e) {
Log.d(TAG, "Error setting camera preview: " + e.getMessage());
}
}
private void getCameraOptimalVideoSize() {
try {
Camera.Parameters parameters = mCamera.getParameters();
List<Camera.Size> mSupportedPreviewSizes = parameters.getSupportedPreviewSizes();
List<Camera.Size> mSupportedVideoSizes = parameters.getSupportedVideoSizes();
Camera.Size optimalSize = CameraHelper.getOptimalVideoSize(mSupportedVideoSizes,
mSupportedPreviewSizes, getWidth(), getHeight());
mOptVideoWidth = optimalSize.width;
mOptVideoHeight = optimalSize.height;
Log.d(TAG, "prepareVideoRecorder: optimalSize:" + mOptVideoWidth + ", " + mOptVideoHeight);
} catch (Exception e) {
Log.e(TAG, "getCameraOptimalVideoSize: ", e);
}
}@Override
public void surfaceDestroyed(SurfaceHolder holder) {
mHolder.removeCallback(this);
mCamera.setPreviewCallback(null);
mCamera.stopPreview();
mCamera.release();
mCamera = null;
}@Override
public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) {
}
}

3、在Fragment中显示摄像头预览

预置一个FrameLayout,实例化一个CameraPreview添加进去

/**
* 视频录制界面
* Created by Rust on 2018/5/17.
*/

public class VideoRecordFragment extends Fragment {
private static final String TAG = "rustAppVideoFrag";
private Button mCaptureBtn;
private CameraPreview mCameraPreview;
public static VideoRecordFragment newInstance() {
return new VideoRecordFragment();
}@Override
public void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.d(TAG, "frag onCreate");
}@Nullable@Override
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
Log.d(TAG, "frag onCreateView");
return inflater.inflate(R.layout.frag_video_record, container, false);
}@Override
public void onViewCreated(@NonNull View view, @Nullable Bundle savedInstanceState) {
Log.d(TAG, "frag onViewCreated");
super.onViewCreated(view, savedInstanceState);
mCaptureBtn = view.findViewById(R.id.capture_btn);//mCaptureBtn.setOnClickListener(mOnClickListener);// 录制键
mCameraPreview = new CameraPreview(getContext());
FrameLayout preview = view.findViewById(R.id.camera_preview);
preview.addView(mCameraPreview);
}
}

使用MediaRecorder录制
给MediaRecorder指定参数后,调用start()开始录制,stop()结束录制

录制开始前,获取camera,mCamera.unlock()解锁;录制完毕后,清除MediaRecorder,mCamera.lock()

private MediaRecorder mMediaRecorder;
public boolean startRecording() {
if (prepareVideoRecorder()) {
mMediaRecorder.start();
return true;
} else {
releaseMediaRecorder();
}
return false;
}
public void stopRecording() {
if (mMediaRecorder != null) {
mMediaRecorder.stop();
}
releaseMediaRecorder();
}
public boolean isRecording() {
return mMediaRecorder != null;
}
private boolean prepareVideoRecorder() {
if (null == mCamera) {
mCamera = getCameraInstance();
}
mMediaRecorder = new MediaRecorder();
mCamera.unlock();
mMediaRecorder.setCamera(mCamera);
mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.DEFAULT);
mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
mMediaRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH));
mMediaRecorder.setVideoSize(mOptVideoWidth, mOptVideoHeight);
mMediaRecorder.setOutputFile(getOutputMediaFile(MEDIA_TYPE_VIDEO).toString());
mMediaRecorder.setPreviewDisplay(mHolder.getSurface());
try {
mMediaRecorder.prepare();
} catch (IllegalStateException e) {
Log.d(TAG, "IllegalStateException preparing MediaRecorder: " + e.getMessage());
releaseMediaRecorder();
return false;
} catch (IOException e) {
Log.d(TAG, "IOException preparing MediaRecorder: " + e.getMessage());
releaseMediaRecorder();
return false;
}
return true;
}
private void releaseMediaRecorder() {
if (mMediaRecorder != null) {
mMediaRecorder.reset();
mMediaRecorder.release();
mMediaRecorder = null;
mCamera.lock();
}
}
private File getOutputMediaFile(int type) {
File mediaStorageDir = new File(Environment.getExternalStoragePublicDirectory(
Environment.DIRECTORY_PICTURES), TAG);
if (!mediaStorageDir.exists()) {
if (!mediaStorageDir.mkdirs()) {
Log.d(TAG, "failed to create directory");
return null;
}
}
String timeStamp = new SimpleDateFormat("yyyyMMdd_HHmmss", Locale.CHINA).format(new Date());
File mediaFile;
if (type == MEDIA_TYPE_IMAGE) {
mediaFile = new File(mediaStorageDir.getPath() + File.separator +
"IMG_" + timeStamp + ".jpg");
outputMediaFileType = "image/*";
} else if (type == MEDIA_TYPE_VIDEO) {
mediaFile = new File(mediaStorageDir.getPath() + File.separator +
"VID_" + timeStamp + ".mp4");
outputMediaFileType = "video/*";
} else {
return null;
}
outputMediaFileUri = Uri.fromFile(mediaFile);
return mediaFile;
}

后台返回时预览黑屏的问题
CameraPreview是我们在Fragment创建时实例化并添加进去的。
应用退到后台后,CameraPreview已经被销毁。应用回到前台时,我们应该在onResume方法中进行操作。恢复CameraPreview。

在Fragment中,判断销毁和重建预览的时机。

@Override
public void onPause() {
super.onPause();
Log.d(TAG, "onPause: 销毁预览");
mCameraPreview = null;
}@Override
public void onResume() {
super.onResume();
Log.d(TAG, "onResume: 回到前台");
if (null == mCameraPreview) {
initCameraPreview();
}
}
private void initCameraPreview() {
mCameraPreview = new CameraPreview(getContext());
FrameLayout preview = mRoot.findViewById(R.id.camera_preview);
preview.addView(mCameraPreview);
}

原创作者:CHSmile,原文链接:https://www.jianshu.com/p/b6386ba20d08

欢迎关注我的微信公众号「码农突围」,分享Python、Java、大数据、机器学习、人工智能等技术,关注码农技术提升•职场突围•思维跃迁,20万+码农成长充电第一站,陪有梦想的你一起成长。

android mediarecorder 输出到流_音视频的采集、编码、封包成 mp4 输出相关推荐

  1. FFMPEG音视频同步-音频实时采集编码封装成MP4

    https://blog.csdn.net/quange_style/article/details/90083173

  2. FFMPEG音视频同步-音视频实时采集并编码推流

    FFMPEG音视频同步-音视频实时采集并编码推流 //------------------------------------------------------------------------- ...

  3. android音视频工程师,音视频学习 (十三) Android 中通过 FFmpeg 命令对音视频编辑处理(已开源)...

    ## 音视频学习 (十三) Android 中通过 FFmpeg 命令对音视频编辑处理(已开源) ## 视音频编辑器 ## 前言 有时候我们想对音视频进行加工处理,比如视频编辑.添加字幕.裁剪等功能处 ...

  4. 音视频编解码 -- 编码参数 CRF

    之前多多少少接触过一些编解码参数,CRF 参数也用过,但是最近在和朋友们聊天时,说到使用 FFMPEG 过程中碰到 CRF 参数,以及具体作用流程,这个之前一直没有跟踪过,也没有详细记录过,所以吊起了 ...

  5. FFmpeg基础:获取音视频的各种编码参数

    文章目录 获取视频编码参数 获取音频编码参数 上一篇文章中介绍了音视频的各种编码参数的概念,这里介绍一下如何通过ffmpeg库获取一个视频文件的各种音视频编码参数.在对视频文件进行处理和转码的时候这些 ...

  6. 「Python|音视频处理|场景案例」如何使用ffmpeg下载m3u8视频到本地并保存成mp4

    本文主要介绍如何使用ffmpeg批量下载视频到本地并保存成指定格式. 文章目录 场景说明 解决方案 源代码 场景说明 当我们希望将网页上的视频下载到本地的时候,我们可能获取到的视频地址是指向.m3u8 ...

  7. (二)音视频:MediaCodec编码桌面信息 完整Demo 进一步理解H264

    (一)音视频:解码H264文件流程 渲染和拿到解码后源数据YUV 完整Demo] (二)音视频:MediaCodec编码桌面信息 完整Demo 进一步理解H264 (三)音视频:解析H264 SPS ...

  8. linux avi 转mp4,怎么把视频avi格式怎么转换成mp4格式

    随着时代的发展,现在的人们已经进入了快餐式的节奏了.但是在这种背景下如不去改变那么则会非常疲惫,所以就会出现大量的电影.电影也就是视频的文件类型多种多样.mp4.mkv.flv.avi等视频格式耳熟能 ...

  9. 视频mkv格式怎么转换成mp4,简单实用的步骤

    视频mkv格式怎么转换成mp4? 前几天有个朋友抱怨说,自己好不容易下载了自己喜欢的电影,结果是mkv格式的,根本播放不了.我相信类似这样的问题,还有很多人经常遇到,那下载到mkv格式的视频,不能播放 ...

最新文章

  1. arm云教室服务器_成都凌点科技告诉你ARM集群服务器适合的应用场景有哪些
  2. 微信小程序之redirectTo、switchTab和navigateTo
  3. 盘点 Python 高手都写不出来的几个错误
  4. java c 引用区别_Java的引用c++的引用和C指针的区别
  5. Hibernate - HHH000352: Unable to release batch statement
  6. Psam_ISO7816
  7. 跟小静读CLR via C#(15)--String,熟悉而又陌生
  8. 排序算法之十 希尔排序(C++版本)
  9. 华为鸿蒙系统老手机能用吗_华为发布鸿蒙2.0手机开发者测试版!华为老手机可申请公测...
  10. html下载文件和上传文件(图片)(java后台(HttpServlet))打开保存路径和选择文件录取+(乱码UTF-8)+包...
  11. python cgi模块 失败_python cgi 连接 sqlite3 失败的问题
  12. 对抗样本生成算法之DeepFool算法
  13. js实现oss批量下载文件_前端实现批量打包下载文件
  14. 《快学BigData》--Linux 常用命令
  15. 第二周学习前端总结与感悟(一)
  16. 《月下独酌》赏析-(李白明月姑娘之一)
  17. 简单测试服务器磁盘读写速度
  18. 【字节跳动实习面经(测试开发岗 一面)四个字:破涕为笑】
  19. pc端/web端/移动端
  20. mysql怎么定位错误信息_如何快速定位MySQL 的错误日志(Error Log)?

热门文章

  1. sql server重命名_在Linux上SQL Server中重命名逻辑和物理文件名
  2. 【转载】Spring Cloud底层原理
  3. [转载] .NET 中可以有类似 JVM 的幻像引用吗?
  4. 【题解】CF#960 H-Santa's Gift
  5. input 输入速度和方向判断、搜索功能的延迟请求
  6. BZOJ 1041 数学
  7. css表示屏幕宽度和高度
  8. Memcached安装及配置
  9. android 照片旋转并保存
  10. 新浪微博 API 使用入门