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

做音频合成的时候找到的一个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.hqwc.cn/news/321969.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

灸哥问答:数据结构对软件开发的作用

在软件开发的浩瀚海洋中&#xff0c;数据结构如同一座坚固的灯塔&#xff0c;为开发者指明方向&#xff0c;确保他们在构建复杂系统时不会迷失。数据结构不仅仅是编程的基础&#xff0c;更是高效、稳定、可扩展软件的核心。 一、提升算法效率 数据结构与算法紧密相连&#xf…

莫比乌斯函数

积性函数定义 若gcd(p,q)1&#xff0c;有f(p*q)f(p)*f(q)&#xff0c;则f(x)是积性函数 其中规定f(1)1&#xff0c;对于积性函数有&#xff1a;所有的积性函数都可以用筛法求出 常见的积性函数有欧拉函数和莫比乌斯函数 筛法求莫比乌斯函数 const int N 1e9 5; const int …

【STM32】STM32学习笔记-编码器接口测速(20)

00. 目录 文章目录 00. 目录01. 预留02. 编码器测速接线图03. 编码器测速程序示例04. 程序下载05. 附录 01. 预留 02. 编码器测速接线图 03. 编码器测速程序示例 Encoder.h #ifndef __ENCODER_H #define __ENCODER_Hvoid Encoder_Init(void); int16_t Encoder_Get(void);#en…

lumina电源维修CCPF-3800/1000-900P-SW Univ Mnt

维修产品有&#xff1a;靶材电源维修&#xff1b;等离子电源维修&#xff1b;镀膜机电源维修&#xff1b;射频电源维修&#xff1b;ICP电源维修&#xff1b;镀膜机控制器维修&#xff1b;RF电源维修&#xff1b;匹配器维修&#xff1b;电源控制器维修&#xff1b;氧化源控制器维…

万界星空科技MES系统中的生产管理

MES系统能够帮助企业实现生产计划管理、生产过程控制、产品质量管理、车间库存管理、项目看板管理等&#xff0c;提高企业制造执行能力。 万界星空MES系统特点&#xff1a; 1. 采用强大数据采集引擎、整合数据采集渠道&#xff08;RFID、条码设备、PLC、Sensor、IPC、PC等&…

16|连接数据库:通过链和代理查询鲜花信息

16&#xff5c;连接数据库&#xff1a;通过链和代理查询鲜花信息 新的数据库查询范式 下面这个图&#xff0c;非常清晰地解释了这个以 LLM 为驱动引擎&#xff0c;从自然语言的&#xff08;模糊&#xff09;询问&#xff0c;到自然语言的查询结果输出的流程。 这种范式结合了…

elementui loading自定义图标和字体样式

需求&#xff1a;页面是用了很多个loading&#xff0c;需要其中有一个字体大些&#xff08;具体到图标也一样的方法&#xff0c;换下类名就行&#xff09; 遇见的问题&#xff1a;改不好的话会影响其他的loading样式&#xff08;一起改变了&#xff09; 效果展示 改之前 改之…

C++完成Query执行sql语句的接口封装和测试

1、在LXMysql.h 创建Query执行函数 //封装 执行sql语句 if sqllen 0 strlen获取字符长度bool Query(const char*sql,unsigned long sqllen0); 2、在LXMysql.cpp编写函数 bool LXMysql::Query(const char* sql, unsigned long sqllen){if (!mysql)//如果mysql没有初始化好{c…

知识图谱与云计算

内容来自B站视频 复旦 肖仰华 老师的讲座&#xff0c;记在这里&#xff0c;不然一会就忘了。 https://www.bilibili.com/video/BV1HG4y1h7zK/?p5&spm_id_frompageDriver 智能的发展是由感知到认知&#xff0c;当下需要发展机器的认知能力。 实现认知智能需要人工智能的很…

01第一个Mybatis程序+引入Junit+引入日志文件logback

Mybatis MyBatis本质上就是对JDBC的封装&#xff0c;通过MyBatis完成CRUD。而对于JDBC&#xff0c;SQL语句写死在Java程序中&#xff0c;不灵活。改SQL的话就要改Java代码。违背开闭原则OCP。对于事务机制&#xff0c;MyBatis支持 或managed模式&#xff0c;JDBC模式中MyBatis…

2023-我的CSDN创作之旅

1.博客内容与数量 2023年共发表博客59篇&#xff0c;内容主要集中在GIS&#xff0c;空间分析等领域 主要内容有&#xff1a; networkx学习 Geospatial Data Science Geocomputation ESDA in PySal SHAP Spatial Data Analysis BikeDNA 以下是对这几个章节主要内容的简…

你的网站或许不需要前端构建(二)

前一阵&#xff0c;有朋友问我&#xff0c;能否在不进行前端编译构建的情况下&#xff0c;用现代语法开发网站界面。 于是&#xff0c;就有了这篇文章中提到的方案。 写在前面 这篇文章&#xff0c;依旧不想讨论构建或不构建&#xff0c;哪一种方案对开发更友好&#xff0c;…