利用 FFmpeg 将 MP4 转成 FLV

最近做一个小项目,要在线播放录制的 MP4 视频,想开源的 flash player 或 html 5 可以播放。可,虽然 MP4 是 H.264 编码,但就是播放不了。可能是封装方式(PS 方式)不一样吧。由于录制用的第三方设备,不能修改参数,只能自己使用工具转码了。

FFmpeg

网上一搜索,就找到了大名鼎鼎的 FFmpeg ,好像 google 的 youtube 后台也是用的这个转码,国内的很多视频播放器核心也是这个。有了这个实现起来就非常简单了。FFmpeg 转码时占用 CPU 很高,可以到 100%,也不知道该怎么解决此问题。转码只要一条指令就行了:

ffmpeg.exe -i source.mp4 -c:v libx264 -crf 24 destination.flv

这是最简单的设置,更多可以去官网看详细的参数,其中 -crf 很重要,是控制转码后视频的质量,质量越高,文件也就越大。

The range of the quantizer scale is 0-51: where 0 is lossless, 23 is default, and 51 is worst possible. A lower value is a higher quality and a subjectively sane range is 18-28. Consider 18 to be visually lossless or nearly so: it should look the same or nearly the same as the input but it isn't technically lossless.

官网的解释(翻译):

此值的范围是 0 到 51:0 表示高清无损;23 是默认值(如果没有指定此参数);51 虽然文件最小,但效果是最差的。

值越小,质量越高,但文件也越大,建议的值范围是 18 到 28。而值 18 是视觉上看起来无损或接近无损的,当然不代表是数据(技术上)的转码无损。

Coding:

实现起来也是很简单,只要使用 Processs 后台转码就行,请看 FfmpegHelper :

 1 using System;
 2 using System.Configuration;
 3 using System.IO;
 4 // reference https://github.com/LeafDuan/WebPrint/tree/master/WebPrint.Framework
 5 using WebPrint.Framework;
 6 // reference https://github.com/LeafDuan/WebPrint/tree/master/WebPrint.Logging
 7 using WebPrint.Logging;
 8 
 9 namespace WebPrint.CameraServer.Helper
10 {
11     public class FfmpegHelper
12     {
13         private static readonly ILogger Logger = LoggerHelper.GetLogger(typeof (FfmpegHelper));
14 
15         private static string Ffmpeg
16         {
17             get { return ConfigurationManager.AppSettings["ffmepg"]; }
18         }
19 
20         private static string Args
21         {
22             get { return ConfigurationManager.AppSettings["args"]; }
23         }
24 
25         private static string FlvPath
26         {
27             get { return ConfigurationManager.AppSettings["flv"]; }
28         }
29 
30         public static string DecodeMp4ToFlv(string mp4, int timeout = 0)
31         {
32             var ffmpeg = "\"{0}\"".Formatting(Ffmpeg);
33             var flv = Path.Combine(FlvPath, (Path.GetFileNameWithoutExtension(mp4) ?? string.Empty) + ".flv");
34             var args = Args.Formatting("\"{0}\"".Formatting(mp4), "\"{0}\"".Formatting(flv));
35             string output, error;
36             if (timeout <= 0)
37                 timeout = 5*60*1000; // timeout = 5 minutes
38             ProcessHelper.Process(ffmpeg, args, timeout, out output, out error);
39             if (!error.IsNullOrEmpty())
40             {
41                 Logger.Error("{0}{1} : {2}{0}".Formatting(Environment.NewLine, "FFmpeg", error));
42             }
43 
44             return flv;
45         }
46     }
47 }
View Code

倒是其中 Process 的实现需要技巧,尤其是针对 output、error 和 timeout 的处理。如果不使用 AutoResetEvent ,process 很容易卡死在 error output 上(IO blocked)。其中针对超时,做了一个处理,就是 kill 掉 process ,免得引起资源霸占和泄露(过多 ffmpeg 进程)。

附:Flash 推荐

可以转码成 html 5 支持的 H.264,也可以其他格式,如 flv。为了兼容 IE6 及以上浏览器,只能使用 flash 播放的方式了。使用的是开源的 vcastr22.swf,可能由于开源,项目现在没有人维护了。

最后吐槽一句:盼 IE9 以下的版本早日寿终正寝。为了随窗口以 16:9 的尺寸自动缩放,兼容 IE6、7 的 css 和 js 是写得累死了。因为非专业前端,找资料都累死了。

感谢

感谢 @eflay 的解决思路,“通过转封装的方式,以复制的效率实现MP4转FLV”。本人不是很懂这些视频的编码,因此没有想到这么好的解决方式。

同时也找到了一篇很好的解决这个问题的文章 h264格式的flv和mkv无损转换成mp4的方法

posted @ 2015-04-03 17:56  Leaf.Duan  阅读(10654)  评论(5编辑  收藏  举报