JavaCV的摄像头实战之六:保存为mp4文件(有声音)

欢迎访问我的GitHub

https://github.com/zq2599/blog_demos

内容:所有原创文章分类汇总及配套源码,涉及Java、Docker、Kubernetes、DevOPS等;

本篇概览

  • 本文是《JavaCV的摄像头实战》的第六篇,在《JavaCV的摄像头实战之三:保存为mp4文件》一文中,咱们将摄像头的内容录制为mp4文件,相信聪明的您一定觉察到了一缕瑕疵:没有声音
  • 虽然《JavaCV的摄像头实战》系列的主题是摄像头处理,但显然音视频健全才是最常见的情况,因此就在本篇补全前文的不足吧:编码实现摄像头和麦克风的录制

关于音频的采集和录制

  • 本篇的代码是在《JavaCV的摄像头实战之三:保存为mp4文件》源码的基础上增加音频处理部分
  • 编码前,咱们先来分析一下,增加音频处理后具体的代码逻辑会有哪些变化
  • 只保存视频的操作,与保存音频相比,步骤的区别如下图所示,深色块就是新增的操作:

JavaCV的摄像头实战之六:保存为mp4文件(有声音)
  • 相对的,在应用结束时,释放所有资源的时候,音视频的操作也比只有视频时要多一些,如下图所示,深色就是释放音频相关资源的操作:

JavaCV的摄像头实战之六:保存为mp4文件(有声音)
  • 为了让代码简洁一些,我将音频相关的处理都放在名为AudioService的类中,也就是说上面两幅图的深色部分的代码都在AudioService.java中,主程序使用此类来完成音频处理
  • 接下来开始编码

开发音频处理类AudioService

  • 首先是刚才提到的AudioService.java,主要内容就是前面图中深色块的功能,有几处要注意的地方稍后会提到:
package com.bolingcavalry.grabpush.extend;

import lombok.extern.slf4j.Slf4j;
import org.bytedeco.ffmpeg.global.avcodec;
import org.bytedeco.javacv.FFmpegFrameRecorder;
import org.bytedeco.javacv.FrameRecorder;
import javax.sound.sampled.AudioFormat;
import javax.sound.sampled.AudioSystem;
import javax.sound.sampled.DataLine;
import javax.sound.sampled.TargetDataLine;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.nio.ShortBuffer;
import java.util.concurrent.ScheduledThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

/**
 * @author willzhao
 * @version 1.0
 * @description 音频相关的服务
 * @date 2021/12/3 8:09
 */
@Slf4j
public class AudioService {

    // 采样率
    private final static int SAMPLE_RATE = 44100;

    // 音频通道数,2表示立体声
    private final static int CHANNEL_NUM = 2;

    // 帧录制器
    private FFmpegFrameRecorder recorder;

    // 定时器
    private ScheduledThreadPoolExecutor sampleTask;

    // 目标数据线,音频数据从这里获取
    private TargetDataLine line;

    // 该数组用于保存从数据线中取得的音频数据
    byte[] audioBytes;

    // 定时任务的线程中会读此变量,而改变此变量的值是在主线程中,因此要用volatile保持可见性
    private volatile boolean isFinish = false;

    /**
     * 帧录制器的音频参数设置
     * @param recorder
     * @throws Exception
     */
    public void setRecorderParams(FrameRecorder recorder) throws Exception {
        this.recorder = (FFmpegFrameRecorder)recorder;

        // 码率恒定
        recorder.setAudioOption("crf", "0");
        // 最高音质
        recorder.setAudioQuality(0);
        // 192 Kbps
        recorder.setAudioBitrate(192000);

        // 采样率
        recorder.setSampleRate(SAMPLE_RATE);

        // 立体声
        recorder.setAudioChannels(2);
        // 编码器
        recorder.setAudioCodec(avcodec.AV_CODEC_ID_AAC);
    }

    /**
     * 音频采样对象的初始化
     * @throws Exception
     */
    public void initSampleService() throws Exception {
        // 音频格式的参数
        AudioFormat audioFormat = new AudioFormat(SAMPLE_RATE, 16, CHANNEL_NUM, true, false);

        // 获取数据线所需的参数
        DataLine.Info dataLineInfo = new DataLine.Info(TargetDataLine.class, audioFormat);

        // 从音频捕获设备取得其数据的数据线,之后的音频数据就从该数据线中获取
        line = (TargetDataLine)AudioSystem.getLine(dataLineInfo);

        line.open(audioFormat);

        // 数据线与音频数据的IO建立联系
        line.start();

        // 每次取得的原始数据大小
        final int audioBufferSize = SAMPLE_RATE * CHANNEL_NUM;

        // 初始化数组,用于暂存原始音频采样数据
        audioBytes = new byte[audioBufferSize];

        // 创建一个定时任务,任务的内容是定时做音频采样,再把采样数据交给帧录制器处理
        sampleTask = new ScheduledThreadPoolExecutor(1);
    }

    /**
     * 程序结束前,释放音频相关的资源
     */
    public void releaseOutputResource() {
        // 结束的标志,避免采样的代码在whlie循环中不退出
        isFinish = true;
        // 结束定时任务
        sampleTask.shutdown();
        // 停止数据线
        line.stop();
        // 关闭数据线
        line.close();
    }

    /**
     * 启动定时任务,每秒执行一次,采集音频数据给帧录制器
     * @param frameRate
     */
    public void startSample(double frameRate) {

        // 启动定时任务,每秒执行一次,采集音频数据给帧录制器
        sampleTask.scheduleAtFixedRate((Runnable) new Runnable() {
            @Override
            public void run() {
                try
                {
                    int nBytesRead = 0;

                    while (nBytesRead == 0 && !isFinish) {
                        // 音频数据是从数据线中取得的
                        nBytesRead = line.read(audioBytes, 0, line.available());
                    }

                    // 如果nBytesRead
  • 上述代码中,有两处要注意:

  • 重点关注recorder.recordSamples,该方法将音频存入了mp4文件

  • 定时任务是在一个新线程中执行的,因此当主线程结束录制后,需要中断定时任务中的while循环,因此新增了volatile类型的变量isFinish,帮助定时任务中的代码判断是否立即结束while循环

改造原本只存视频的代码

  • 接着是对《JavaCV的摄像头实战之三:保存为mp4文件》一文中RecordCameraSaveMp4.java的改造,为了不影响之前章节在github上的代码,这里我新增了一个类RecordCameraSaveMp4WithAudio.java,内容与RecordCameraSaveMp4.java一模一样,接下来咱们来改造这个RecordCameraSaveMp4WithAudio类
  • 先增加AudioService类型的成员变量:
    // 音频服务类
    private AudioService audioService = new AudioService();
  • 接下来是关键,initOutput方法负责帧录制器的初始化,现在要加上音频相关的初始化操作,并且还要启动定时任务去采集和处理音频,如下所示,AudioService的三个方法都在此调用了,注意定时任务的启动要放在帧录制器初始化之后:
    @Override
    protected void initOutput() throws Exception {
        // 实例化FFmpegFrameRecorder
        recorder = new FFmpegFrameRecorder(RECORD_FILE_PATH,        // 存放文件的位置
                                           getCameraImageWidth(),   // 分辨率的宽,与视频源一致
                                           getCameraImageHeight(),  // 分辨率的高,与视频源一致
                                            0);                      // 音频通道,0表示无

        // 文件格式
        recorder.setFormat("mp4");

        // 帧率与抓取器一致
        recorder.setFrameRate(getFrameRate());

        // 编码格式
        recorder.setPixelFormat(AV_PIX_FMT_YUV420P);

        // 编码器类型
        recorder.setVideoCodec(avcodec.AV_CODEC_ID_MPEG4);

        // 视频质量,0表示无损
        recorder.setVideoQuality(0);

        // 设置帧录制器的音频相关参数
        audioService.setRecorderParams(recorder);

        // 音频采样相关的初始化操作
        audioService.initSampleService();

        // 初始化
        recorder.start();

        // 启动定时任务,采集音频帧给帧录制器
        audioService.startSample(getFrameRate());
  • output方法保存原样,只处理视频帧(音频处理在定时任务中)
    @Override
    protected void output(Frame frame) throws Exception {
        // 存盘
        recorder.record(frame);
    }
  • 释放资源的方法中,增加了音频资源释放的操作:
    @Override
    protected void releaseOutputResource() throws Exception {
        // 执行音频服务的资源释放操作
        audioService.releaseOutputResource();

        // 关闭帧录制器
        recorder.close();
    }
  • 至此,将摄像头视频和麦克风音频存为mp4文件的功能已开发完成,再写上main方法,注意参数30表示抓取和录制的操作执行30秒,注意,这是程序执行的时长, 不是录制视频的时长
    public static void main(String[] args) {
        // 录制30秒视频
        new RecordCameraSaveMp4WithAudio().action(30);
    }
  • 运行main方法,等到控制台输出下图红框的内容时,表示视频录制完成:

JavaCV的摄像头实战之六:保存为mp4文件(有声音)
  • 打开mp4文件所在目录,如下图,红框中就是刚刚生成的文件和相关信息,注意蓝框的内容,证明该文件包含了视频和音频的数据:

JavaCV的摄像头实战之六:保存为mp4文件(有声音)
  • 用VLC播放验证,结果视频和声音都正常
  • 至此,咱们已完成了保存音视频文件的功能,得益于JavaCV的强大,整个过程是如此的轻松愉快,接下来请继续关注欣宸原创,《JavaCV的摄像头实战》系列还会呈现更多丰富的应用;

源码下载

名称 链接 备注 项目主页
https://github.com/zq2599/blog_demos

该项目在GitHub上的主页 git仓库地址(https)
https://github.com/zq2599/blog_demos.git

该项目源码的仓库地址,https协议 git仓库地址(ssh) git@github.com:zq2599/blog_demos.git 该项目源码的仓库地址,ssh协议

  • 这个git项目中有多个文件夹,本篇的源码在javacv-tutorials文件夹下,如下图红框所示:

JavaCV的摄像头实战之六:保存为mp4文件(有声音)
  • javacv-tutorials里面有多个子工程,《JavaCV的摄像头实战》系列的代码在 simple-grab-push工程下:

JavaCV的摄像头实战之六:保存为mp4文件(有声音)

你不孤单,欣宸原创一路相伴

搜索「程序员欣宸」,我是欣宸,期待与您一同畅游Java世界…

https://github.com/zq2599/blog_demos

Original: https://www.cnblogs.com/bolingcavalry/p/15877663.html
Author: 程序员欣宸
Title: JavaCV的摄像头实战之六:保存为mp4文件(有声音)

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/588822/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • 开篇说几句

    一直一览都是博客园的粉丝,从中查询了很多资料,给了我很多的帮助,随着时间的积累,自己有一些感悟和学习笔记,今天申请了个博客吧,吧自己的一些想法写出来,保留份记忆。 Original…

    Java 2023年6月8日
    072
  • Java 之 String 类型

    因为对象的默认值是null,所以String的默认值也是null;但它又是一种特殊的对象,有其它对象没有的一些特性。 首先String不属于8种基本数据类型(byte, char,…

    Java 2023年5月29日
    064
  • 使用Java8 Stream流的skip + limit实现批处理

    1、一般进行批处理时会将数据加入到一个临时的集合中,当数据量达到一定大小后进行下一步操作,数据量不足时需要进行额外的判断; 2、若使用Java8的Stream流中的 skip + …

    Java 2023年5月29日
    048
  • 程序猿学习抖音短视频制作

    @ 前言 一、短视频简介: 1.简介 1.1 短视频变现方式 1.2 例子 1.3 短视频优势 (1)⻔槛低有利于传播 (2)喜欢看短视频的人更多 1.4 变现方式举例 (1)打造…

    Java 2023年6月13日
    060
  • 在ASP.NET 中调用RSACryptoServiceProvider失败,提示未找到文件

    CspParameters RSAParams = new CspParameters(); RSAParams.Flags = CspProviderFlags.UseMachi…

    Java 2023年6月14日
    055
  • Spring Cloud GateWay 简单示例

    前提:提供一个注册中心,可以使用Eureka Server。供gateway转发请求时获取服务实例。 一、新建GateWay项目 1、引入maven依赖 org.springfra…

    Java 2023年5月30日
    065
  • 频频曝出程序员被抓,我们该如何避免面向监狱编程?

    过去几年,频频有程序员被抓的消息刷爆技术圈,无论是技术高手、公司合伙人还是普通程序员,甚至整个科技公司,都有因为违法被抓的案例。 案例 比如: 1、著名的Nignx之父Igor S…

    Java 2023年5月29日
    073
  • 进程 线程 协程

    黑书 计算机操作系统中说了很多。 首先,在一个cpu的情况下,进程是一个程序运行时的总和,一个时刻肯定只有一个进程在执行,只是cpu会使用进程调度算法来回切换进程而已,就绪,运行,…

    Java 2023年5月30日
    075
  • MySQL中的事务和MVCC

    本篇博客参考掘金小册——MySQL 是怎样运行的:从根儿上理解 MySQL 以及极客时间——MySQL实战45讲。 虽然我们不是DBA,可能对数据库没那么了解,但是对于数据库中的索…

    Java 2023年6月5日
    093
  • 我是怎么入行做风控的

    引言 常听到周围有人说”风控”这个词,只知道这是一个神秘的部门,对他们做的事却一知半解,只知道这个风控部门对公司非常重要,任何活动和信息都最好向风控部门报备…

    Java 2023年6月15日
    094
  • Springboot学习

    具体内容: 包含核心基础、Web原理、单元测试、数据访问、指标监控等章节 SpringBoot 官方文档 https://www.cnblogs.com/youcoding/p/1…

    Java 2023年5月30日
    097
  • 12.安装证书

    import java.io.BufferedReader;import java.io.File;import java.io.FileInputStream;import ja…

    Java 2023年6月13日
    058
  • JAVA获取文件的路径方法(获取classpath路径)

    ClassLoader 提供了两个方法用于从装载的类路径中取得资源: public URL getResource (String name); public InputStrea…

    Java 2023年5月29日
    079
  • [推荐]MyBatis 核心技术与面试 34 讲

    MyBatis 核心技术与面试 34 讲 职业生涯中常被问到: 如何成为某方面的高手? 如何快速搞定某项技术? 我现在的水平处于什么阶段? …… 我暗暗想,…

    Java 2023年5月30日
    0202
  • Java问题排解

    1. 新安装的jdk11,竟然发现没有自动安装jre,运行程序不能运行。 2. 关于eclipse Installed JREs 添加 jdk10.0.2时,提示”Ta…

    Java 2023年5月29日
    064
  • 一篇不太正经的个人介绍

    你好啊,这里是程序员田同学。 在网上胡写乱画也快半年了,公众号上也连更了一个多月,新关注的朋友就比较疑惑,这家伙是谁呀?整天又在写些什么?为啥要关注他呀? 今天有必要写一篇文章介绍…

    Java 2023年6月8日
    059
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球