- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章PHP调用ffmpeg对视频截图并拼接脚本由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
PHP脚本调用ffmpeg对视频截图并拼接,供大家参考,具体内容如下 。
目前支持MKV,MPG,MP4等常见格式的视频,其他格式有待测试 。
12P 一张截图平均生成时间 1.64s 100个视频,大概需要2分半左右 。
9P 一张截图平均生成时间 1.13s 100个视频,大概需要2分钟左右 。
6P 一张截图平均生成时间 0.86s 100个视频,大概需要1分半左右 。
3P 一张截图平均生成时间 0.54s 100个视频,大概需要1分钟左右 。
- <?php
- define('DS', DIRECTORY_SEPARATOR);
- date_default_timezone_set("Asia/Shanghai");
- class FileLoader
- {
- //路径变量
- private $rootdir = '';
- private $tmp = "tmp"; //tmp 目录
- private $source = "mpg"; //source 目录
- private $destination = "screenshoot"; //目标截图路径
- private $emptyImageName = "empty.jpg"; //合成的背景图
- //文件数组
- private $maxShoots = 12; //最大的截图数
- private $videoInfo = NULL;
- private $files = array(); //文件数
- private $fileArray = array();
- private $extensionArray = array("mpg","mkv","mp4","avi","3gp","mov"); //支持的格式
- private $timeArray = array("00:00:10","00:00:20","00:00:30","00:01:00","00:01:30","00:02:00","00:02:30","00:03:00","00:03:30","00:03:40","00:03:50","00:04:00");
- //统计变量
- private $timeStart = 0;
- private $timeEnd = 0;
- private $fileCount = 0;
- private $successCount = 0;
- private $failedCount = 0;
- /**
- * 初始化信息
- */
- function __construct()
- {
- file_put_contents("log.txt","");
- $this->rootdir = dirname(__FILE__);
- $count = count($this->timeArray);
- for($i=1;$i<=$count;$i++)
- {
- $ii = $i-1;
- $this->fileArray[$ii] = $this->tmp.DS.$i.".jpg";
- }
- }
- /**
- * 当前时间,精确到小数点
- */
- private static function microtime_float()
- {
- list($usec, $sec)= explode(" ", microtime());
- return ((float)$usec + (float)$sec);
- }
- /**
- * 00:00:00 时间转秒
- */
- private static function timeToSec($time)
- {
- $p = explode(':',$time);
- $c = count($p);
- if ($c>1)
- {
- $hour = intval($p[0]);
- $minute = intval($p[1]);
- $sec = intval($p[2]);
- }
- else
- {
- throw new Exception('error time format');
- }
- $secs = $hour * 3600 + $minute * 60 + $sec;
- return $secs;
- }
- /**
- * 00:00:00 时间转秒
- */
- private static function secToTime($time)
- {
- $hour = floor($time/3600);
- $min = floor(($time - $hour * 3600)/60);
- $sec = $time % 60;
- $timeStr = sprintf("%02d:%02d:%02d",$hour,$min,$sec);
- return $timeStr;
- }
- /**
- * 获取全部文件
- */
- private function getFiles($dir)
- {
- $files = array();
- $dir = rtrim($dir, "/\\") . DS;
- $dh = opendir($dir);
- if ($dh == false) { return $files; }
- while (($file = readdir($dh)) != false)
- {
- if ($file{0} == '.') { continue; }
- $path = $dir . $file;
- if (is_dir($path))
- {
- $files = array_merge($files, $this->getFiles($path));
- }
- elseif (is_file($path))
- {
- $files[] = $path;
- }
- }
- closedir($dh);
- return $files;
- }
- /**
- * 搜索路径
- */
- public function searchDir($sourcePath = NULL)
- {
- $this->timeStart = $this->microtime_float();
- if ($sourcePath)
- {
- $this->rootdir = $sourcePath;
- }
- if (file_exists($this->rootdir) && is_dir($this->rootdir))
- {
- $this->files = $this->getFiles($this->rootdir.DS.$this->source);
- }
- $this->fileCount = count($this->files);
- foreach ($this->files as $path)
- {
- $fi = pathinfo($path);
- $flag = array_search(strtolower($fi['extension']),$this->extensionArray);
- if (!$flag) continue;
- $this->getScreenShoot(basename($path));
- }
- $this->timeEnd = $this->microtime_float();
- $time = $this->timeEnd - $this->timeStart;
- if($this->fileCount > 0)
- {
- $str = sprintf("[TOTAL]: Cost Time:%8s | Total File:[%d] | Successed:[%d] | Failed:[%d] | Speed:%.2fs per file\n",$this->secToTime($time),$this->fileCount,$this->successCount,$this->failedCount,$time/$this->fileCount);
- file_put_contents("log.txt",$str,FILE_APPEND);
- }
- else
- {
- $str = sprintf("[TOTAL]: Cost Time:%8s | Total File:[%d] | Successed:[%d] | Failed:[%d] | Speed:%.2fs per file\n",$this->secToTime($time),$this->fileCount,$this->successCount,$this->failedCount,0);
- file_put_contents("log.txt",$str,FILE_APPEND);
- }
- }
- /**
- * 获取视频信息
- */
- private function getVideoInfo($file){
- $re = array();
- exec(".".DS."ffmpeg -i {$file} 2>&1", $re);
- $info = implode("\n", $re);
- if(preg_match("/No such file or directory/i", $info))
- {
- return false;
- }
- if(preg_match("/Invalid data/i", $info)){
- return false;
- }
- $match = array();
- preg_match("/\d{2,}x\d+/", $info, $match);
- list($width, $height) = explode("x", $match[0]);
- $match = array();
- preg_match("/Duration:(.*?),/", $info, $match);
- if($match)
- {
- $duration = date("H:i:s", strtotime($match[1]));
- }else
- {
- $duration = NULL;
- }
- $match = array();
- preg_match("/bitrate:(.*kb\/s)/", $info, $match);
- $bitrate = $match[1];
- if(!$width && !$height && !$duration && !$bitrate){
- return false;
- }else{
- return array(
- "file" => $file,
- "width" => $width,
- "height" => $height,
- "duration" => $duration,
- "bitrate" => $bitrate,
- "secends" => $this->timeToSec($duration)
- );
- }
- }
- /**
- * 设置截图时间
- */
- private function setShootSecends($secends,$useDefault = NO)
- {
- if($useDefault)
- {
- if($secends<18)
- {
- $time = 1;
- }else
- {
- $time = 5;
- }
- $range = floor(($secends - $time)/ ($this->maxShoots));
- if ($range < 1)
- {
- $range = 1;
- }
- $this->timeArray = array();
- for($i=0;$i<$this->maxShoots;$i++)
- {
- $this->timeArray[$i] = $this->secToTime($time);
- $time = $time + $range;
- if ($time > $secends) break;
- }
- }
- }
- /**
- * 拼接图片
- */
- private function getFixedPhoto($fileName)
- {
- $target = $this->rootdir.DS.$this->emptyImageName;//背景图片
- $target_img = Imagecreatefromjpeg($target);
- $source= array();
- foreach ($this->fileArray as $k=>$v)
- {
- $source[$k]['source'] = Imagecreatefromjpeg($v);
- $source[$k]['size'] = getimagesize($v);
- }
- $tmpx=5;
- $tmpy=5;//图片之间的间距
- for ($i=0; $i< count($this->timeArray); $i++)
- {
- imagecopy($target_img,$source[$i]['source'],$tmpx,$tmpy,0,0,$source[$i]['size'][0],$source[$i]['size'][1]);
- $target_img = $this->setTimeLabel($target_img,$tmpx,$tmpy,$source[$i]['size'][0],$source[$i]['size'][1],$this->timeArray[$i]);
- $tmpx = $tmpx+ $source[$i]['size'][0];
- $tmpx = $tmpx+5;
- if(($i+1) %3 == 0){
- $tmpy = $tmpy+$source[$i]['size'][1];
- $tmpy = $tmpy+5;
- $tmpx=5;
- }
- }
- $target_img = $this->setVideoInfoLabel($target_img,$tmpx,$tmpy,$this->videoInfo);
- Imagejpeg($target_img,$this->rootdir.DS.$this->destination.DS.$fileName.'.jpg');
- }
- /**
- * 设置时间刻度标签
- */
- private function setTimeLabel($image,$image_x,$image_y,$image_w,$image_h,$img_text)
- {
- imagealphablending($image,true);
- //设定颜色
- $color=imagecolorallocate($image,255,255,255);
- $ttf_im=imagettfbbox(30 ,0,"Arial.ttf",$this->img_text);
- $w = $ttf_im[2] - $ttf_im[6];
- $h = $ttf_im[3] - $ttf_im[7];
- unset($ttf_im);
- $txt_y =$image_y+$image_h+$h-5;
- $txt_x =$image_x+$w+5;
- imagettftext($image,30,0,$txt_x,$txt_y,$color,"Arial.ttf",$img_text);
- return $image;
- }
- /**
- * 设置视频信息标签
- */
- private function setVideoInfoLabel($image,$txt_x,$txt_y,$videoInfo)
- {
- imagealphablending($image,true);
- $color=imagecolorallocate($image,0,0,0);
- imagettftext($image,32,0,100,2000+30,$color,"FZLTHJW.ttf","FileName:".basename($videoInfo["file"]));
- imagettftext($image,32,0,1600,2000+30,$color,"Arial.ttf","Size:".$videoInfo["width"]."x".$videoInfo["height"]);
- imagettftext($image,32,0,100,2000+120,$color,"Arial.ttf","Duration:".$videoInfo["duration"]);
- imagettftext($image,32,0,1600,2000+120,$color,"Arial.ttf","Bitrate:".$videoInfo["bitrate"]);
- return $image;
- }
- /**
- * 屏幕截图
- */
- public function getScreenShoot($fileName)
- {
- $fi = pathinfo($fileName);
- $this->videoInfo = $this->getVideoInfo($this->rootdir.DS.$this->source.DS.$fileName);
- if($this->videoInfo)
- {
- $this->setShootSecends($this->videoInfo["secends"]);
- for ($i=0; $i< count($this->timeArray); $i++ )
- {
- $cmd=".".DS."ffmpeg -ss ". $this->timeArray[$i] ." -i ". $this->rootdir.DS.$this->source.DS.$fileName ." -y -f image2 -s 720*480 -vframes 1 ".$this->rootdir.DS.$this->fileArray[$i];
- exec($cmd,$out,$status);
- }
- $this->getFixedPhoto($fileName);
- $str = sprintf("[%s]:OK...........[%s][%2dP]%-30s\n",date("y-m-d h:i:s",time()),$this->videoInfo["duration"],count($this->timeArray),$fileName);
- file_put_contents("log.txt",$str,FILE_APPEND);
- $this->successCount += 1;
- }else
- {
- $str = sprintf("[%s]:FAILED.................................[%s][%2dP]%-30s\n",date("y-m-d h:i:s",time()),$this->videoInfo["duration"],count($this->timeArray),$fileName);
- file_put_contents("log.txt",$str,FILE_APPEND);
- $this->failedCount += 1;
- }
- }
- /**
- * TODO:
- * 截取图片,
- * 需要配置ffmpeg-php,比较麻烦,
- * 但是这个类确实挺好用的。
- */
- public function getScreenShoot2($fileName)
- {
- if(extension_loaded('ffmpeg')){//判断ffmpeg是否载入
- $mov = new ffmpeg_movie($this->rootdir.DS.$this->source.DS.$fileName);//视频的路径
- $count = $mov->getFrameCount();
- $ff_frame = $mov->getFrame(floor($count/2));
- if($ff_frame)
- {
- $gd_image = $ff_frame->toGDImage();
- $img=$this->rootdir.DS."test.jpg";//要生成图片的绝对路径
- imagejpeg($gd_image, $img);//创建jpg图像
- imagedestroy($gd_image);//销毁一图像
- }
- }else{
- echo "ffmpeg没有载入";
- }
- }
- }
- $fileLoader = new FileLoader();
- $fileLoader->searchDir();
- ?>
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我.
最后此篇关于PHP调用ffmpeg对视频截图并拼接脚本的文章就讲到这里了,如果你想了解更多关于PHP调用ffmpeg对视频截图并拼接脚本的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我对此很陌生,我在这里的论坛上检查过答案,但我没有找到任何真正可以帮助我的答案。我正在尝试播放 res/raw 文件夹中的视频。到目前为止我已经设置了这段代码: MediaPlayer mp; @Ov
我可以播放一个视频剪辑,检测视频的结尾,然后创建一个表单,然后播放另一个视频剪辑。我的问题是,表单 react 不正确,我创建了带有提交按钮和两个单选按钮可供选择的表单。我希望让用户进行选择,验证响应
首先,我必须说我在web2py讨论组中看到过类似的内容,但我不太理解。 我使用 web2py 设置了一个数据库驱动的网站,其中的条目只是 HTML 文本。其中大多数将包含 img和/或video指向相
我正在尝试在视频 View 中播放 YouTube 视频。 我将 xml 布局如下: 代码是这样的: setContentView(R.layout.webview); VideoV
我正在开发一个需要嵌入其中的 youtube 视频播放器的 android 应用程序。我成功地从 API 获得了 RTSP 视频 URL,但是当我试图在我的 android 视频 View 中加载这个
我目前正在从事一个使用 YouTube API 的网络项目。 我完全不熟悉 API。所以每一行代码都需要付出很多努力。 使用以下代码,我可以成功检索播放列表中的项目: https://www.goog
是否可以仅使用视频 ID 和 key 使用 API V3 删除 youtube 视频?我不断收到有关“必需参数:部分”丢失的错误消息。我用服务器和浏览器 api 键试了一下这是我的代码: // $yo
所以我一直坚持这个大约一个小时左右,我就是无法让它工作。到目前为止,我一直在尝试从字符串中提取整个链接,但现在我觉得只获取视频 ID 可能更容易。 RegEx 需要从以下链接样式中获取 ID/URL,
var app = angular.module('speakout', []).config( function($sceDelegateProvider) {
我正在努力从 RSS 提要中阅读音频、视频新闻。我如何确定该 rss 是用于新闻阅读器还是用于音频或视频? 这是视频源:http://feeds.cbsnews.com/CBSNewsVideo 这是
利用python反转图片/视频 准备:一张图片/一段视频 python库:pillow,moviepy 安装库 ?
我希望在用户双击视频区域时让我的视频全屏显示,而不仅仅是在他们单击控件中的小图标时。有没有办法添加事件或其他东西来控制用户点击视频时发生的情况? 谢谢! 最佳答案 按照 Musa 的建议,附
关闭。这个问题需要更多 focused .它目前不接受答案。 想改进这个问题?更新问题,使其仅关注一个问题 editing this post . 7年前关闭。 Improve this questi
我有一个公司培训视频加载到本地服务器上。我正在使用 HTML5 的视频播放来观看这些视频。该服务器无法访问网络,但我已加载 apache 并且端口 8080 对同一网络上的所有机器开放。 这些文件位于
我想混合来自 video.mp4 的视频(时长 1 分钟)和来自 audio.mp3 的音频(10 分钟持续时间)到一个持续时间为 1 分钟的输出文件中。来自 audio.mp3 的音频应该是从 4
关闭。这个问题需要更多 focused .它目前不接受答案。 想改进这个问题?更新问题,使其仅关注一个问题 editing this post . 8年前关闭。 Improve this questi
我正在尝试使用 peer/getUserMedia 创建一个视频 session 网络应用程序。 目前,当我将唯一 ID 发送到视频 session 时,我能够听到/看到任何加入我的 session
考虑到一段时间内的观看次数,我正在评估一种针对半自动脚本的不同方法,该脚本将对视频元数据执行操作。 简而言之,只要视频达到指标中的某个阈值,就说观看次数,它将触发某些操作。 现在要执行此操作,我将不得
我正在通过iBooks创建专门为iPad创建动态ePub电子书的网站。 它需要支持youtube视频播放,所以当我知道视频的直接路径时,我正在使用html5 标记。 有没有一种使用html5 标签嵌入
我对Android不熟悉,我想浏览youtube.com并在Webview内从网站显示视频。当前,当我尝试执行此操作时,将出现设备的浏览器,并让我使用设备浏览器浏览该站点。如果Webview不具备这种
我是一名优秀的程序员,十分优秀!