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分钟左右

<div class="jb51code">
<pre class="brush:php;">
<?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",0); 
  file_put_contents("log.txt",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:(.*?),/",$match);
    if($match)
    {
    $duration = date("H:i:s",strtotime($match[1]));
    }else
    {
    $duration = NULL;
    }

    $match = array();
    preg_match("/bitrate:(.*kb\/s)/",$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,$source[$i]['size'][0],$source[$i]['size'][1]); 
  $target_img = $this->setTimeLabel($target_img,$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,$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);
    $ttf_im=imagettfbbox(30,"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,$txt_x,$txt_y,$color,$img_text);
    return $image;
    }

/**

  • 设置视频信息标签
    */
    private function setVideoInfoLabel($image,$videoInfo)
    {
    imagealphablending($image,true);

    $color=imagecolorallocate($image,0);

    imagettftext($image,32,100,2000+30,"FZLTHJW.ttf","FileName:".basename($videoInfo["file"]));
    imagettftext($image,1600,"Size:".$videoInfo["width"]."x".$videoInfo["height"]);
    imagettftext($image,2000+120,"Duration:".$videoInfo["duration"]);
    imagettftext($image,"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",FILE_APPEND);
    $this->successCount += 1;
    }else
    {
    $str = sprintf("[%s]:FAILED.................................[%s][%2dP]%-30s\n",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();
?>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程之家。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


极简概括: PHP 的开源内存缓存扩展,类比Redis,但是一般都用Redis,所以APCu用的很少。官方文档:https://www.php.net/manual/zh/apcu.configuration.php解决问题:类比Redis做缓存组件,提升性能,同步数据使用。
请看如下代码: $list = [1,2,4,5]; $list2 = [5,6,7,9]; foreach ($list as $key =&gt; &amp;$value) { $value = strval($value); } foreach ($list2 as $key =&gt; $v
一、nginx 安装 1. 在nginx官网下载nginx源码&#x9;提供一个nginx官网下载地址: http://nginx.org/download/nginx-1.12.2.tar.gz 注意:请先确认 是否已经安装过 pcre pcre-devel openssl openssl-devel z
先看图 数据条数:9000+ 第1,2行,golangʾxcelize方式导出,耗时:5s 第3行,PHP+xlswriter方式导出,耗时:2min 一、介绍 xlswriter是一个高效处理excel文件的PHP扩展,底层以C语言实现;处理速度是PHPExcel几十倍甚至几百倍的效率。 官方链
今天使用Thinkphp5做异步任务传递where参数时遇到一个问题: 有一段如下代码: $where[&#39;jst.supplier&#39;] = [&#39;exp&#39;, Db::raw(&#39;&gt;0 or jst.is_supplier=1&#39;)]; 在使用swool
汇总 PHP5.1: autoload PDO MySQLi 类型约束 PHP5.2: JSON 支持 PHP5.3: 命名空间 匿名函数 闭包 新增魔术方法__callStatic()和__invoke() 新增魔术变量__DIR__ 动态调用静态方法 延迟静态绑定 Heredoc和 Nowdoc
文章浏览阅读8.4k次,点赞8次,收藏7次。SourceCodester Online Tours & Travels Management System pay.php sql injectionLine 16 of pay.php invokes a SQL query built using unvalidated input. This call could allow an attacker to modify the statement’s meaning or to execute arbitrary SQL commands.SQL
文章浏览阅读3.4k次,点赞46次,收藏51次。本文为大家介绍在windwos系统搭建typecho博客+cpolar内网穿透工具将博客发布到公共网络环境,实现远程也可以访问和操作。_windows搭建typecho
文章浏览阅读1.1k次。- php是最优秀, 最原生的模板语言, 替代语法,让php更加的优雅的与html生活在一起 -->请放心, 最终生成的,或者说用户最终看到的,仍然是一个html文档, php代码中的内容不会被泄漏的。-- 将php与html代码混编的时候,大括号很容易造成配对错误,最好杜绝它 -->php标签内部代码由php.exe解释, php标签之外的代码原样输出,仍由web服务器解析。-- 所以php的流程控制语句, 都提供了替代语法,用冒号代替大括号 -->php echo '百变鹏仔'?_利用php将静态页面修改为动态页面
文章浏览阅读1.1k次,点赞18次,收藏15次。整理K8s网络相关笔记博文内容涉及 Linux network namespace 认知以及彼此通信Demo,实际中的应用理解不足小伙伴帮忙指正不必太纠结于当下,也不必太忧虑未来,当你经历过一些事情的时候,眼前的风景已经和从前不一样了。——村上春树。_linux network namespace 多端通信 模式认知
文章浏览阅读1.2k次,点赞22次,收藏19次。此网络模型提供了一个逻辑二层(L2)网络,该网络封装在跨 Kubernetes 集群节点的现有三层(L3)网络拓扑上。使用此模型,可以为容器提供一个隔离的 L2 网络,而无需分发路由。封装网络带来了少量的处理开销以及由于覆盖封装生成 IP header 造成的 IP 包大小增加。封装信息由 Kubernetes worker 之间的 UDP 端口分发,交换如何访问 MAC 地址的网络控制平面信息。此类网络模型中常用的封装是 VXLAN、Internet 协议安全性 (IPSec) 和 IP-in-IP。_k8s网络组件对比
文章浏览阅读1.1k次,点赞14次,收藏19次。当我们谈论网络安全时,我们正在讨论的是保护我们的在线空间,这是我们所有人的共享责任。网络安全涉及保护我们的信息,防止被未经授权的人访问、披露、破坏或修改。
文章浏览阅读1.3w次,点赞3次,收藏7次。尽管您可以通过 ping 命令解析出网站的 IP 地址,但是可能在浏览器中访问时仍然遇到问题,这可能是因为浏览器使用的 DNS 解析结果不同于 ping 命令使用的解析结果。可能是因为您的网络或设备上设置了防火墙,阻止了对特定网站的访问。有些国家或组织可能会对特定的域名进行屏蔽,从而阻止访问相关网站。如果您的网络使用代理服务器进行访问控制,可能会由于代理服务器的配置问题导致无法访问某些网站。即使您的网络和设备一切正常,目标网站本身可能也存在问题,例如服务器故障、维护或过载,导致无法访问。_能ping通打不开网页
文章浏览阅读839次,点赞22次,收藏19次。本系统带文档lw万字以上文末可领取本课题的JAVA源码参考。
文章浏览阅读2.1k次,点赞31次,收藏22次。基于微信小程序奶茶点餐外卖系统设计与实现(PHP后台+Mysql)可行性分析毕设源代码毕业设计,数据安全和系统稳定性以及团队能力和资源配备方面都具备较好的条件。因此,该项目的可行性较高。:黄菊华老师《Vue.js入门与商城开发实战》《微信小程序商城开发》图书作者,CSDN博客专家,在线教育专家,CSDN钻石讲师;微信小程序作为一种快捷、方便的移动应用形式,成为很多用户点餐外卖的首选。项目的界面和功能都可以定制,包安装运行!项目配有对应开发文档、开题报告、任务书、PPT、论文模版等。
文章浏览阅读1.8k次,点赞52次,收藏38次。本文主要通过对系统的前台系统和后台管理系统进行了功能性需求分析,对系统的安全性和可扩展性进行了非功能性需求分析。在详细的需求分析的基础上,根据系统的功能设计确定了数据库结构,实现完整的代码编写。Lucky+Baby母婴用品网站使用 Dreamweaver、HBuilder代码编辑器、Apache服务器等开发工具,完成了系统的主要模块的页面设计和功能实现。本文展示了首页页面的实现效果图,并通过代码和页面介绍了用户注册功能、商品搜索功能、生成订单和查看我的订单功能、在线付款功能功能的实现过程。
文章浏览阅读1.5k次,点赞45次,收藏40次。本设计主要实现集人性化、高效率、便捷等优点于一身的人事信息管理系统,完成首页、系统用户、通知公告、部门信息、员工薪资、考勤签到、员工请假、招聘信息、应聘信息等功能模块。
文章浏览阅读1k次。该错误通常出现在数据库读取结果集数据时,比如当我们写好SQL语句从数据库读取数据时,本身应该返回结果集,再给结果集中读取数据。解决思路:这种错误一般是因为echo后面输出了一个数组导致的,或者是数组作为字符串进行拼接运算时导致的。该错误直译为:警告:mysqli_fetch_assoc函数期望参数1是mysqli的结果集,但是给了一个布尔值。这种错误是PHP解析器在解析时遇到了语法错误,直译为:解析错误:语法错误,意料之外的...该错误直译为:提示:未定义的索引:username。_array to string conversion in
文章浏览阅读2.7w次。解决http请求报错context deadline exceeded (Client.Timeout exceeded while awaiting headers)_context deadline exceeded (client.timeout exceeded while awaiting headers)
文章浏览阅读1.3k次,点赞26次,收藏24次。复杂网络是一种由大量相互连接的元素(节点或顶点)组成的网络结构,这些连接通常是非常复杂和动态的。这些网络可以在各种领域中发现,包括社交网络、生物学系统、信息技术和交通系统等。_代理建模