使用 PHP-FFMpeg 操作视频/音频文件

news2025/1/12 23:15:07

做音频合成的时候找到的一个php操作ffmpeg 的类库。GitHub地址:https://github.com/PHP-FFMpeg/PHP-FFMpeg/。本文的例子大部分都是上面的

在使用之前请安装好 FFMpeg 。如何安装?请看 FFmpeg 安装教程。

使用composer快速安装 > composer require php-ffmpeg/php-ffmpeg。

注意:请在 php.ini 中开启这两个函数proc_open,proc_get_status。找到 disable_functions 将里面的这两个函数去掉就行了

目录说明

/usr/loca/bin ffmpeg 的执行目录

/mnt/hgfs/www/test 本文章的测试目录

这里主要用 1080.mp4 这个视频做测试,下面就是这个 18 秒的视频(chrome 谷歌浏览器不显示播放器是因为https 的站不能使用未加密的http资源,唉)

https://www.bilibili.com/video/av17244788/

使用时请配置 ffmpeg,ffprobe 的执行文件绝对路径。我定义了一些测试用的视频音频文件

$path = [
    'ffmpeg.binaries'  => '/usr/local/bin/avconv',
    'ffmpeg.binaries' => '/usr/local/bin/ffmpeg',
    'ffprobe.binaries' => '/usr/local/bin/avprobe',
    'ffprobe.binaries' => '/usr/local/bin/ffprobe',
];
$ffmpeg = FFMpeg\FFMpeg::create($path);
$a1 = '/mnt/hgfs/www/test/a1.mp3';
$v1 = '/mnt/hgfs/www/test/v1.mp4';
$v2 = '/mnt/hgfs/www/test/v2.mp4';
$v3 = '/mnt/hgfs/www/test/v3.mp4';
$v1080 = '/mnt/hgfs/www/test/1080.mp4';

1、拼接视频/音频

$newFile = '/mnt/hgfs/www/test/video.mp4';
$video = $ffmpeg->open($v1);
$video->concat(array($v1,$v2,$v3))->saveFromSameCodecs($newFile, TRUE);

若是已存在合成的新文件 ($newFile 已存在),将会报错,请确保同一目录下不存在相同的文件

2、提取图像

提取一张

$video = $ffmpeg->open($v1080);
$frame = $video->frame(FFMpeg\Coordinate\TimeCode::fromSeconds(2));//提取第几秒的图像
$frame->save('image.jpg');

抽取多张

$video = $ffmpeg->open($v1080);
$video->filters()
    ->extractMultipleFrames(FFMpeg\Filters\Video\ExtractMultipleFramesFilter::FRAMERATE_EVERY_SEC, '/mnt/hgfs/www/test/image/')
    ->synchronize();
 
$video->save(new FFMpeg\Format\Video\X264('libfdk_aac'), '/mnt/hgfs/www/test/v2new.mp4');

注意:此方法会抽取对应的图片帧,而且每次都提取 400 张,不知道是不是我参数设置的问题(试了 FRAMERATE_EVERY_SEC,FRAMERATE_EVERY_2SEC,FRAMERATE_EVERY_10SEC)

3、生成音频波形

$audio = $ffmpeg->open($a1);
$waveform = $audio->waveform(640, 120, array('#00FF00'));
$waveform->save('waveform.png');//必须保存为 png 格式

若要提取视频的音频波形,须先转换为音频

// Open your video file
$video = $ffmpeg->open( 'video.mp4' );
 
// Set an audio format
$audio_format = new FFMpeg\Format\Audio\Mp3();
 
// Extract the audio into a new file as mp3
$video->save($audio_format, 'audio.mp3');
// Set the audio file
$audio = $ffmpeg->open( 'audio.mp3' );
 
// Create the waveform
$waveform = $audio->waveform();
$waveform->save( 'waveform.png' );

【免费分享】音视频学习资料包、大厂面试题、技术视频和学习路线图,资料包括(C/C++,Linux,FFmpeg webRTC rtmp hls rtsp ffplay srs 等等)有需要的可以点击788280672加群免费领取~

4、调整视频大小

$video = $ffmpeg->open($v1080);
$video->filters()->resize(new FFMpeg\Coordinate\Dimension(200,400), FFMpeg\Filters\Video\ResizeFilter::RESIZEMODE_FIT, true);
$video->save(new FFMpeg\Format\Video\X264('libfdk_aac'), '/mnt/hgfs/www/test/v1080_new.mp4');

注意:x264 类默认使用 libfaac 为编码器,但我安装的是 libfdk_aac,所以这里要指定为 libfdk_aac,不然会报错

参数说明:

resize(Dimension $dimension, $mode = ResizeFilter::RESIZEMODE_FIT, $forceStandards = true)

$dimension 调整后的视频宽高

$mode 四种缩放模式

RESIZEMODE_FIT 按给定值调整

RESIZEMODE_INSET 在给定的尺寸内调整大小,可能是按宽为基准(高等比缩放),也可能是按高为基准(宽等比缩放)

RESIZEMODE_SCALE_WIDTH 高为给定值,宽按比例缩放

RESIZEMODE_SCALE_HEIGHT 宽为给定值,高按比例缩放

$forceStandards ture / false,是否强制使用最近的纵横比标准


5、视频添加水印

$video = $ffmpeg->open($v1080);
$watermarkPath = '/mnt/hgfs/www/test/water.png';
$absolute = ['x' => 50,'y' => 100];
$relative = [
    'position' => 'relative',
    'bottom' => 50,
    'right' => 50
 ];
$video->filters()->watermark($watermarkPath, $absolute);
$video->save(new FFMpeg\Format\Video\X264('libfdk_aac'), '/mnt/hgfs/www/test/1080_new.mp4');

watermark($imagePath, array $coordinates = array())

$imagePath 水印图片路径

$coordinates 水印坐标

position 可选项 relative(相对定位) / absolute(绝对,默认)

若为 relative,有四个参数可选,top 、bottom、left、right ,分别对应四个方位

在上面的例子中就是在视频的右下角,距离右边50,距离下边50 的位置处添加水印(这个位置坐标是水印图片的左下角位置),如图

若为 absolute,直接填写 x 和 y 坐标即可 ,如图

6、调整视频的帧率

关于 帧率 和 GOP 介绍 http://blog.csdn.net/xiangjai/article/details/44238005

这玩意儿我也不懂,设置了几个值进行转换,但都time out 了,1g的虚拟机玩不起

$video = $ffmpeg->open($v1080);
$video->filters()->framerate(new \FFMpeg\Coordinate\FrameRate(3000), 120);
$video->save(new FFMpeg\Format\Video\X264('libfdk_aac'), '/mnt/hgfs/www/test/1080_new.mp4');

7、截取视频/音频

$video = $ffmpeg->open($v1080);
$video->filters()->clip(FFMpeg\Coordinate\TimeCode::fromSeconds(10));
$video->save(new FFMpeg\Format\Video\X264('libfdk_aac'), '/mnt/hgfs/www/test/1080_new.mp4');

clip($start, $duration = null)

从第 $start 秒开始,取 $duration 秒,若 $duration 不填,则截取至最后

8、裁剪视频

$video = $ffmpeg->open($v1080);
$video->filters()->crop(new FFMpeg\Coordinate\Point("t*100", 0, true), new FFMpeg\Coordinate\Dimension(960, 540));
$video->save(new FFMpeg\Format\Video\X264('libfdk_aac'), '/mnt/hgfs/www/test/1080_new.mp4');

crop(Point $point, Dimension $dimension)

$dimension 为裁剪后的尺寸

Point($x, $y, $dynamic = false)

$x 和 $y 为裁剪的起始坐标,$dynamic 为是否动态裁剪

动态裁剪是什么意思呢,比如:

Point("t*100", 0, true) 裁剪出来的视频就是画面从左边动态的移动到右边,然后就固定在左边

100 为速度,值越大,移动速度越快。话说这个动态裁剪没啥用的感觉

下面的视频为上面代码裁剪的,对比一下有什么不同

https://www.bilibili.com/video/av17244824/

9、音频转换

$audio = $ffmpeg->open($a1);
$format = new FFMpeg\Format\Audio\Flac();
$format->on('progress', function ($audio, $format, $percentage) {
    echo "$percentage % 进度";
});
$format->setAudioChannels(2)->setAudioKiloBitrate(256);
$audio->save($format, 'a1.flac');

Flac 为无损压缩格式

setAudioChannels 声道设置,1单声道,2双声道,3立体声

setAudioKiloBitrate 比特率

11、音频添加元数据

$audio = $ffmpeg->open($a1);
$audio->filters()->addMetadata([
    "title" => "Test Title",
    "artist" => "Jam00 artist",
    "album" => "Test album",
    "composer" => "Jam00",
    "track" => 1,
    "year" => 2017,
    "description" => "jam00 test description",
]);
$audio->save(new \FFMpeg\Format\Audio\Mp3, 'a1_new.mp3');

目前支持的数据是 title(标题),artist(艺术家),album(专辑),artist(艺术家),composer(作曲家),track(轨道),year(年),description(描述),artwork(艺术作品)

注:FFmpeg(3.2.2版本)只支持MP3文件添加 artwork 元数据

使用格式工厂查看a1_new.mp3的元数据

...
Format                                   : MPEG Audio
File size                                : 1.43 MiB
Duration                                 : 1 min 33 s
Overall bit rate mode                    : Constant
Overall bit rate                         : 128 kb/s
Album                                    : Test album
Track name                               : Test Title
Track name/Position                      : 1
Performer                                : Jam00 artist
Composer                                 : Jam00
Writing library                          : LAME3.99.5
year                                     : 2017
description                              : jam00 test description
...

12、Frame 提取图像

$video = $ffmpeg->open($v1080);
$frame = new FFMpeg\Media\Frame($video, FFMpeg\Driver\FFMpegDriver::load($path), FFMpeg\FFProbe::create($path), FFMpeg\Coordinate\TimeCode::fromSeconds(10));
$frame->save('frame.jpg');

其实 例子2 提取视频图像的方法 frame 调用的就是 Frame 类

13、从视频中提取动图

$video = $ffmpeg->open($v1080);
$video->gif(FFMpeg\Coordinate\TimeCode::fromSeconds(10), new FFMpeg\Coordinate\Dimension(400, 200), 3)->save('1080.gif');

gif(TimeCode $at, Dimension $dimension, $duration = null)

从第 $at 秒开始提取,持续 $duration 秒,保存为 $dimension指定大小(下面的例子为400x200) 的gif图

若不设置 $duration ,将会得到一个静止的gif图

动图太大,我就不上传了

14、视频格式转换

$video = $ffmpeg->open($v1080);
$format = new FFMpeg\Format\Video\X264('libfdk_aac');
$format->setKiloBitrate(1000)->setAudioChannels(2)->setAudioKiloBitrate(256);
$format->on('progress', function ($video, $format, $percentage) {
    echo "$percentage % 进度";
});
$video->save($format, '/mnt/hgfs/www/test/video.avi');

setKiloBitrate 设置视频比特率

setAudioChannels 声道设置,1单声道,2双声道,3立体声

setAudioKiloBitrate 设置音频比特率

15、添加额外参数(若你精通 ffmpeg 命令行参数)

$video = $ffmpeg->open($v1080);
$format = new FFMpeg\Format\Video\X264('libfdk_aac');
$format->setAdditionalParameters(array('foo', 'bar'));
$video->save($format, 'video.avi');

foo / bar 为 ffmpeg 支持的参数,这个就不测了

16、使用 FFProbe 提取元数据

$ffprobe = FFMpeg\FFProbe::create($path);
//视频
$videoInfo = $ffprobe->format($v1080);
//音频
$audioInfo = $ffprobe->format($a1);
//也可以使用 get 获取特定值,第二个参数为默认值(若该参数不存在将返回此默认值)
$duration = $ffprobe->format($v1080)->get('duration',100);
echo "<pre>";
print_r($videoInfo);
print_r($audioInfo);
echo "</pre>";
echo '视频时长:'.$duration;

FFMpeg\FFProbe\DataMapping\Format Object
(
    [properties:FFMpeg\FFProbe\DataMapping\AbstractData:private] => Array
        (
            [filename] => /mnt/hgfs/www/test/1080.mp4
            [nb_streams] => 2
            [nb_programs] => 0
            [format_name] => mov,mp4,m4a,3gp,3g2,mj2
            [format_long_name] => QuickTime / MOV
            [start_time] => 0.000000
            [duration] => 18.882000
            [size] => 9062983
            [bit_rate] => 3839840
            [probe_score] => 100
            [tags] => Array
                (
                    [major_brand] => isom
                    [minor_version] => 512
                    [compatible_brands] => isomiso2mp41
                    [encoder] => Lavf57.41.100
                )
 
        )
 
)
FFMpeg\FFProbe\DataMapping\Format Object
(
    [properties:FFMpeg\FFProbe\DataMapping\AbstractData:private] => Array
        (
            [filename] => /mnt/hgfs/www/test/a1.mp3
            [nb_streams] => 1
            [nb_programs] => 0
            [format_name] => mp3
            [format_long_name] => MP2/3 (MPEG audio layer 2/3)
            [start_time] => 0.000000
            [duration] => 93.348000
            [size] => 186696
            [bit_rate] => 16000
            [probe_score] => 51
        )
 
)
视频时长:18.882000

本内容为博主原创,转载请注明出处。

本文链接使用 PHP-FFMpeg 操作视频/音频文件

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1358841.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

VUE3跳转页面时 定时器未清除解决

一,问题 1、在vue中使用setTimeout定时器的时候&#xff0c;可能会遇到关不掉的情况&#xff0c;会存在明明已经在beforeDestroy和destroyed中设置了定时器清除了&#xff0c;但是有时候没生效&#xff0c;定时器还会继续执行。 2、在这里需要说一下setTimeout的使用场景&…

【React系列】受控非受控组件

本文来自#React系列教程&#xff1a;https://mp.weixin.qq.com/mp/appmsgalbum?__bizMzg5MDAzNzkwNA&actiongetalbum&album_id1566025152667107329) 一. refs 的使用 在React的开发模式中&#xff0c;通常情况下不需要、也不建议直接操作DOM原生&#xff0c;但是某些…

红日靶场第一关stack靶场

安装黑客后台工具&#xff0c;使用cs工具 也就是cobaltstrike工具 首先我们先使用msfvenom工具生成一个用于windows的木马后台文件&#xff0c;然后我们将生成的木马文件名字叫做620.exe 然后我们要使用msfconsole工具监听咱们的木马后门 启动监听 用蚁剑开启我们所设置的木…

商城系统哪家好,如何选择商城系统?

之前我们做电商项目时&#xff0c;通过百度搜索“java商城系统”、”商城系统有哪些“等关键词&#xff0c;发现各类系统琳琅满目&#xff0c;我们先按商城系统推出时间做了一个统计&#xff0c;然后针对各系统进行了一定的调研。 推出时间超过10年的&#xff1a;shop、ecshop…

Java爬虫之Jsoup

1.Jsoup相关概念 Jsoup很多概念和js类似&#xff0c;可参照对比理解 Document &#xff1a;文档对象。每份HTML页面都是一个文档对象&#xff0c;Document 是 jsoup 体系中最顶层的结构。 Element&#xff1a;元素对象。一个 Document 中可以着包含着多个 Element 对象&#…

纯前端上传word,xlsx,ppt,在前端预览并下载成图片(预览效果可以,下载图片效果不太理想)

纯前端上传word,xlsx,ppt,在前端预览并下载成图片&#xff08;预览效果可以&#xff0c;下载图片效果不太理想&#xff09; 一.安装依赖二、主要代码 预览效果链接: https://github.com/501351981/vue-office 插件文档链接: https://501351981.github.io/vue-office/examples/d…

webrtc报文记录

tcp.port 10443 || tcp.port 6080 || udp.port 8000 https://download.csdn.net/download/dualvencsdn/88706745

LeetCode 2397. 被列覆盖的最多行数:二进制枚举

【LetMeFly】2397.被列覆盖的最多行数&#xff1a;二进制枚举 力扣题目链接&#xff1a;https://leetcode.cn/problems/maximum-rows-covered-by-columns/ 给你一个下标从 0 开始的 m x n 二进制矩阵 mat 和一个整数 cols &#xff0c;表示你需要选出的列数。 如果一行中&am…

数据结构和算法-插入排序(算法效率 折半优化 顺序表与链表插入排序 代码实现)

文章目录 插入排序算法实现算法效率分析优化-折半插入排序代码实现对链表进行插入排序小结 插入排序 首先49当作第一个已经排好序得元素&#xff0c;将第二个元素与前面得元素对比&#xff0c;发现小于49&#xff0c;于是49移动位置 此时将65与之前元素对比&#xff0c;发现其…

【C++期末编程题题库】代码+详解18道

适合期末复习c看&#xff0c;或者刚入门c的小白看&#xff0c;有的题会补充知识点&#xff0c;期末复习题的代码一般比较简单&#xff0c;所以语法上没那么严谨。本文所有题目要求全在代码块的最上面。 目录 1、设计复数类 2、设计Computer类 3、实现相加的函数模板 4、圆类…

全网最简单vscode使用Makefile调试多文件的C/C++代码

前言 vscode调试C/C教程很多&#xff0c;操作麻烦&#xff0c;这里试图找到一个最简单的使用vscode调试C/C代码的方法。这里是使用Makefile的多文件方式。 测试文件 tree . ├── func.c ├── func.h ├── main.c └── Makefilefun.c #include <stdio.h> #in…

对低效的会议说“不!”

根据微软对全球 31, 000 名员工开展的一项调查&#xff0c;低效的会议是影响工作效率的第一大干扰因素&#xff0c;其次是召开过多的会议。 大大小小的同步会、讨论会、审查会、复盘会不仅将工作时间拆解得支离破碎&#xff0c;还会让成员因「会议恢复综合症」而无法立即从无效…

Certum与Geotrust的OV多域名证书

Certum和Geotrust都是知名的CA认证机构&#xff0c;旗下的SSL证书产品丰富&#xff0c;有单域名SSL证书、多域名SSL证书以及通配符SSL证书。这些SSL数字证书作为一种重要的网络安全产品&#xff0c;能够实现数据加密和身份验证&#xff0c;保障网站的安全性和隐私性。OV多域名S…

【快速全面掌握 WAMPServer】14.各种组件的升级方法

网管小贾 / sysadm.cc WAMPServer 更新很快&#xff0c;这是件好事&#xff01; 但是 WAMPServer 更新快是因为他很勤劳吗&#xff1f; 其实这个问题的原因并不是出自 WAMPServer 自身&#xff0c;而是来自它的各个组件。 是的&#xff0c;你能想像得到&#xff0c;比如 PHP…

数据结构和算法-数据结构的基本概念和三要素和数据类型和抽象数据类型

文章目录 总览数据结构的基本概念总览数据早期和现代的计算机处理的数据数据元素-描述一个个体数据对象-一类数据元素什么是数据结构小结 数据结构的三要素总览逻辑结构-集合结构逻辑结构-线性结构逻辑结构-树形结构逻辑结构-图形结构逻辑结构-小结数据的运算物理结构&#xff…

Struts2 远程代码执行漏洞S2-001分析

自 Struts2 在 2007 年爆出第一个远程代码执行漏洞 S2-001 以来&#xff0c;在其后续的发展过程中不断爆出更多而且危害更大的远程代码执行漏洞&#xff0c;而造成 Struts2 这么多 RCE 漏洞的主要原因就是 OGNL 表达式。这里以 Struts2 的第一个漏洞 S2-001 为例来对 Struts2 远…

新年启新程 | 开门红!菊风中标重庆三峡银行双录及产品销售可回溯系统项目

INTRODUCTION 近年来&#xff0c;随着人们需求的转变和金融科技的高速发展&#xff0c;银行开始朝着数智化方向转型。为顺应客户行为变迁&#xff0c;银行同业积极构建远程银行云服务生态。同时&#xff0c;面对业务的升级以及新的监管要求&#xff0c;现有音视频功能难以满足…

第三届先进控制、自动化与机器人国际会议(ICACAR 2024) | Ei、Scopus双检索

会议简介 Brief Introduction 2024年第三届先进控制、自动化与机器人国际会议(ICACAR 2024) 会议时间&#xff1a;2024年5月24-26日 召开地点&#xff1a;中国重庆 大会官网&#xff1a;ICACAR 2024-2024 3rd International Conference on Advanced Control, Automation and Ro…

yolov8 tracking编码为web 和 rtsp流输出

1 基础工作 打开cmd 输入 conda env list 输入 conda activate py38 查看 nvidia-smi 查看 nvcc&#xff0c;如下图所示 cuda为11.7 &#xff0c;为确认可以查看program files 下面的cuda 安装&#xff0c;看到11.7 就行了&#xff0c;读者可以自行确认自己的版本。 查看nvid…

第 378 场 LeetCode 周赛题解

A 检查按位或是否存在尾随零 枚举&#xff1a;枚举两个元素的组合即可 class Solution { public:bool hasTrailingZeros(vector<int> &nums) {int n nums.size();for (int i 0; i < n; i)for (int j 0; j < i; j)if ((nums[i] | nums[j]) % 2 0)return tru…