C#/。NET用のソリッドFFmpegラッパー


91

私はしばらくの間、C#/。NET用の堅牢なFFmpegラッパーをWebで検索してきました 。しかし、私はまだ何か役に立つものを思い付いていません。私は次の3つのプロジェクトを見つけましたが、それらはすべてアルファ段階の初期に死んでいるようです。

FFmpeg.NETffmpeg-
シャープな
FFLIB.NET

だから私の質問は、もっと成熟したラッパープロジェクトを誰かが知っているかどうかです。
ジョブキューなどを備えた完全なトランスコーディングエンジンを探していません。単純なラッパーなので、コマンドライン呼び出しを行ってからコンソール出力を解析する必要はありませんが、メソッド呼び出しを行ったり、イベントリスナーを使用して進行したりできます。

また、初期段階であっても、アクティブなプロジェクトについてはお気軽にご連絡ください。



1
これで何か新しいことはありますか?ラッパーは進歩しましたか?
Avi 2011

3
@Lillemandenラッパーをリリースまたはオープンソース化したことはありますか?
ニックベネディクト

興味深いことに、質問はほぼ6年前のものですが、OP(@JacobPoulRichardt)はどの回答も受け入れませんでした。
Ofer Zelig

1
自分で作成したラッパーを使用することになったため、提案されたプロジェクトは使用しませんでした。私はもはやffmpegを使用していないので、戻ってそれらのいずれかを試す時間がありませんでした。しかし、それらをスキミングした後、ほとんどの答えに賛成しました。ですから、どの答えも他の答えよりも「正しい」とは言えないと思います。
Jacob Poul Richardt 2015年

回答:


23

これは私自身のラッパーです:https//github.com/AydinAdn/MediaToolkit

MediaToolkitは次のことができます。

  • ビデオファイルを他のさまざまなビデオ形式に変換します。
  • ビデオトランスコーディングタスクを実行します。
    • 設定可能なオプション:Bit rateFrame rateResolution / sizeAspect ratioDuration of video
  • オーディオトランスコーディングタスクを実行します。
    • 構成可能なオプション: Audio sample rate
  • FILM、PAL、またはNTSCTV規格を使用してビデオを物理フォーマットに変換する
    • 媒体が含まれます:DVDDVDV50VCDSVCD

進行中に更新しています。ご利用いただけます。パッケージマネージャーコンソールを使用してインストールすることもできます。

PM> Install-Package MediaToolkit

ツールキットは、さまざまなビデオクリップとオーディオクリップを特定の出力解像度の1つに多重化/レンダリングできますか?
アントニオペトリカ2014

いいえ、単純な変換を追跡する人々のために使用するように設計されています。とは言うものの、FFmpegが提供しなければならないすべてのことを実行できるv2が間もなく登場します。
アイディン2014

アイディン、ありがとう。この新しいリリースについて知らせてください。
アントニオペトリカ2014

素晴らしく見えます!これまでのところ、お疲れ様でした!
SpoiledTechie.com 2015

ねえアイディン、これも画面を記録できますか?
TEK 2016

14

いくつかのラッパーを試した後、私はこれを使用しました:FFmpegはC#/。NETとMonoの安全でないバインディングを自動生成しました

これは、FFmpeg名前空間内のすべてのクラスの低レベルの相互運用バインディングのセットです。実際のラッパーほど便利ではないかもしれませんが、重要なことをしたい場合は、IMOが.NetでFFmpegを操作するための最良のソリューションです。

長所:

  • 作品
  • 信頼できる-FFMpeg自体を信頼していると仮定すると、バグを導入するサードパーティのラッパーコードはありません。
  • FFmpegの最新バージョンに常に更新されます
  • すべてのバインディング用の単一のnugetパッケージ
  • XMLドキュメントが含まれていますが、オンラインドキュメントのFFmpegドキュメントを引き続き使用できます。

短所:

  • 低レベル:c構造体へのポインターの操作方法を知っている必要があります。
  • それを機能させるには、最初にいくつかの作業が必要です。公式の例から学ぶことをお勧めします。

注:このスレッドはFFmpeg APIの使用に関するものですが、一部のユースケースでは、ffmpeg.exeのコマンドラインインターフェイスを使用するのが最善です。


.Net Framework(コアではない)を対象としたプロジェクトからそれを使用することができましたか?ここで何が欠けているのか
わかり

@YoavFeuersteinはい。
シャチ


10

ASP.NET / Windowsサービス(.NET)アプリケーションのFFmpegを使用しました。しかし、コンソールを解析せずに、コマンドラインを使用することになりました。これを使用することで(私は簡単に制御する方法がありました)、FFmpegの更新と、複数のコアでの複数の変換の実行を行いました。


わかりました、私は似たようなことから始めました。しかし、私は誰かがより良い解決策を持っていることを望んでいます。
Jacob Poul Richardt 2010

4

このnugetパッケージを使用できます:

成熟したプロジェクトについてお伺いしたことは承知しておりますが、期待に応えられるプロジェクトは見当たらないので、自分で作ることにしました。変換を簡単にキューに入れて並列に実行し、メディアをさまざまな形式に変換し、独自の引数をffmpegに送信し、ffmpeg +イベントリスナーからの出力を現在の進行状況で解析できます。

Install-Package Xabe.FFmpeg

使いやすいクロスプラットフォームのFFmpegラッパーを作成しようとしています。

これについての詳細はhttps://xabe.net/product/xabe_ffmpeg/で見つけることができます

詳細はこちら:https//xabe.net/product/xabe_ffmpeg/#documentation

変換は簡単です:

IConversionResult result = await Conversion.ToMp4(Resources.MkvWithAudio, output).Start();

進歩が必要な場合:

IConversion conversion = Conversion.ToMp4(Resources.MkvWithAudio, output);
conversion.OnProgress += (duration, length) => { currentProgress = duration; } 
await conversion.Start();

こんにちは... FFMPEGを使用して、Webページからのストリーミングデータをトランスコードし、RTMPサーバーに送信する必要があります。C#のwinformプログラムにバイトの配列があります。トランスコードしてRTMPサーバーに送信するだけです。このラッパーを使用してそれを行うことはできますか?Linuxでsocketioを使用するnodejsサーバーを使用してこれを行いました。そのプラットフォームでは、stdinを介してバイナリストリームを送信し、stderrで変換ステータスを受け取ります。Xabeラッパーを使用してそれを行うことはできますか?
jstuardo

3

MediaHandlerProというffmpegラッパーライブラリで遊んでいます。

http://www.mediasoftpro.com

これまでのところ有望なようです。


これはどのようにうまくいきましたか?また、その作業を行うプロセスとしてMediaHandlerスポーンしますffmpeg.exeか、それとも実際のP / Invokeライブラリがありますか?
グレンスレイデン2016

私はそれをいくつかのプロジェクトで使用することになりました。負荷の高い本番環境でうまく機能しました。私がそれを使用してからしばらく経ちましたが、私が覚えていることから、はい、それはプロセスとしてffmpeg.exeを生成します。
クリストフチャン

3

私は同じことを研究していて、もともとは変換に最適なMediaToolKit(別の回答で言及)を使用していましたが、今はもう少し堅牢なものが必要です。

成熟していてまだアクティブに見えるオプションの1つは、https//github.com/hudl/HudlFfmpeg です。詳細についてはこちらをご覧ください:http//public.hudl.com/bits/archives/2014/08/15/announcing -hudlffmpeg-ac-framework-to-make-ffmpeg-interaction-simple /

多くの場合に適さない可能性がある別のオプションは、c#コードから直接exeを呼び出すことです:http//www.codeproject.com/Articles/774093/Another-FFmpeg-exe-Csharp-Wrapper


2

1
リンクをありがとうございますが、私が見る限り、あなたはint C#ではなくJavaで書いています。
Jacob Poul Richardt 2010年

こんにちはlillemanden、私が与えたリンクは実際にはJavaで実装されており、記事の下部にあるzipをダウンロードすると、その中にjarアーカイブファイルがあることがわかります。ありがとう、イリヤ
イリヤ

回答のリンクが切れているようです:「このサイトにアクセスできません-ivolo.mit.eduが応答するのに時間がかかりすぎました。」
パン2017年

2

ここに行きます...このコードのほとんどは2年以上前のものであるため、多くの非同期のものが欠落しており、古い命名規則を使用しています。かなり長い間本番環境で実行されています〜JT

internal static class FFMpegArgUtils
    {
        public static string GetEncodeVideoFFMpegArgs(string sSourceFile, MP4Info objMp4Info, double nMbps, int iWidth, int iHeight, bool bIncludeAudio, string sOutputFile)
        {
            //Ensure file contains a video stream, otherwise this command will fail
            if (objMp4Info != null && objMp4Info.VideoStreamCount == 0)
            {
                throw new Exception("FFMpegArgUtils::GetEncodeVideoFFMpegArgs - mp4 does not contain a video stream");
            }

            int iBitRateInKbps = (int)(nMbps * 1000);


            StringBuilder sbArgs = new StringBuilder();
            sbArgs.Append(" -y -threads 2 -i \"" + sSourceFile + "\" -strict -2 "); // 0 tells it to choose how many threads to use

            if (bIncludeAudio == true)
            {
                //sbArgs.Append(" -acodec libmp3lame -ab 96k");
                sbArgs.Append(" -acodec aac -ar 44100 -ab 96k");
            }
            else
            {
                sbArgs.Append(" -an");
            }


            sbArgs.Append(" -vcodec libx264 -level 41 -r 15 -crf 25 -g 15  -keyint_min 45 -bf 0");

            //sbArgs.Append(" -vf pad=" + iWidth + ":" + iHeight + ":" + iVideoOffsetX + ":" + iVideoOffsetY);
            sbArgs.Append(String.Format(" -vf \"scale=iw*min({0}/iw\\,{1}/ih):ih*min({0}/iw\\,{1}/ih),pad={0}:{1}:({0}-iw)/2:({1}-ih)/2\"",iWidth, iHeight));

            //Output File
            sbArgs.Append(" \"" + sOutputFile + "\"");
            return sbArgs.ToString();
        }

        public static string GetEncodeAudioFFMpegArgs(string sSourceFile, string sOutputFile)
        {
            var args = String.Format(" -y -threads 2 -i \"{0}\" -strict -2  -acodec aac -ar 44100 -ab 96k -vn \"{1}\"", sSourceFile, sOutputFile);
            return args;


            //return GetEncodeVideoFFMpegArgs(sSourceFile, null, .2, 854, 480, true, sOutputFile);
            //StringBuilder sbArgs = new StringBuilder();
            //int iWidth = 854;
            //int iHeight = 480;
            //sbArgs.Append(" -y -i \"" + sSourceFile + "\" -strict -2 "); // 0 tells it to choose how many threads to use
            //sbArgs.Append(" -acodec aac -ar 44100 -ab 96k");
            //sbArgs.Append(" -vcodec libx264 -level 41 -r 15 -crf 25 -g 15  -keyint_min 45 -bf 0");
            //sbArgs.Append(String.Format(" -vf \"scale=iw*min({0}/iw\\,{1}/ih):ih*min({0}/iw\\,{1}/ih),pad={0}:{1}:({0}-iw)/2:({1}-ih)/2\"", iWidth, iHeight));
            //sbArgs.Append(" \"" + sOutputFile + "\"");
            //return sbArgs.ToString();
        }
    }

internal class CreateEncodedVideoCommand : ConsoleCommandBase
    {
        public event ProgressEventHandler OnProgressEvent;

        private string _sSourceFile;
        private  string _sOutputFolder;
        private double _nMaxMbps;

        public double BitrateInMbps
        {
            get { return _nMaxMbps; }
        }

        public int BitrateInKbps
        {
            get { return (int)Math.Round(_nMaxMbps * 1000); }
        }

        private int _iOutputWidth;
        private int _iOutputHeight;

        private bool _bIsConverting = false;
        //private TimeSpan _tsDuration;
        private double _nPercentageComplete;
        private string _sOutputFile;
        private string _sOutputFileName;


        private bool _bAudioEnabled = true;
        private string _sFFMpegPath;
        private string _sExePath;
        private string _sArgs;
        private MP4Info _objSourceInfo;
        private string _sOutputExt;

        /// <summary>
        /// Encodes an MP4 to the specs provided, quality is a value from 0 to 1
        /// </summary>
        /// <param name="nQuality">A value from 0 to 1</param>
        /// 
        public CreateEncodedVideoCommand(string sSourceFile, string sOutputFolder, string sFFMpegPath, double nMaxBitrateInMbps, MP4Info objSourceInfo, int iOutputWidth, int iOutputHeight, string sOutputExt)
        {
            _sSourceFile = sSourceFile;
            _sOutputFolder = sOutputFolder;
            _nMaxMbps = nMaxBitrateInMbps;
            _objSourceInfo = objSourceInfo;
            _iOutputWidth = iOutputWidth;
            _iOutputHeight = iOutputHeight;
            _sFFMpegPath = sFFMpegPath;
            _sOutputExt = sOutputExt;
        }

        public void SetOutputFileName(string sOutputFileName)
        {
            _sOutputFileName = sOutputFileName;
        }


        public override void Execute()
        {
            try
            {
                _bIsConverting = false;

                string sFileName = _sOutputFileName != null ? _sOutputFileName : Path.GetFileNameWithoutExtension(_sSourceFile) + "_" + _iOutputWidth + "." + _sOutputExt;
                _sOutputFile = _sOutputFolder + "\\" + sFileName;

                _sExePath = _sFFMpegPath;
                _sArgs = FFMpegArgUtils.GetEncodeVideoFFMpegArgs(_sSourceFile, _objSourceInfo,_nMaxMbps, _iOutputWidth, _iOutputHeight, _bAudioEnabled, _sOutputFile);

                InternalExecute(_sExePath, _sArgs);
            }
            catch (Exception objEx)
            {
                DispatchException(objEx);
            }
        }

        public override string GetCommandInfo()
        {
            StringBuilder sbInfo = new StringBuilder();
            sbInfo.AppendLine("CreateEncodeVideoCommand");
            sbInfo.AppendLine("Exe: " + _sExePath);
            sbInfo.AppendLine("Args: " + _sArgs);
            sbInfo.AppendLine("[ConsoleOutput]");
            sbInfo.Append(ConsoleOutput);
            sbInfo.AppendLine("[ErrorOutput]");
            sbInfo.Append(ErrorOutput);

            return base.GetCommandInfo() + "\n" + sbInfo.ToString();
        }

        protected override void OnInternalCommandComplete(int iExitCode)
        {
            DispatchCommandComplete( iExitCode == 0 ? CommandResultType.Success : CommandResultType.Fail);
        }

        override protected void OnOutputRecieved(object sender, ProcessOutputEventArgs objArgs)
        {
            //FMPEG out always shows as Error
            base.OnOutputRecieved(sender, objArgs);

            if (_bIsConverting == false && objArgs.Data.StartsWith("Press [q] to stop encoding") == true)
            {
                _bIsConverting = true;
            }
            else if (_bIsConverting == true && objArgs.Data.StartsWith("frame=") == true)
            {
                //Capture Progress
                UpdateProgressFromOutputLine(objArgs.Data);
            }
            else if (_bIsConverting == true && _nPercentageComplete > .8 && objArgs.Data.StartsWith("frame=") == false)
            {
                UpdateProgress(1);
                _bIsConverting = false;
            }
        }

        override protected void OnProcessExit(object sender, ProcessExitedEventArgs args)
        {
            _bIsConverting = false;
            base.OnProcessExit(sender, args);
        }

        override public void Abort()
        {
            if (_objCurrentProcessRunner != null)
            {
                //_objCurrentProcessRunner.SendLineToInputStream("q");
                _objCurrentProcessRunner.Dispose();
            }
        }

        #region Helpers

        //private void CaptureSourceDetailsFromOutput()
        //{
        //    String sInputStreamInfoStartLine = _colErrorLines.SingleOrDefault(o => o.StartsWith("Input #0"));
        //    int iStreamInfoStartIndex = _colErrorLines.IndexOf(sInputStreamInfoStartLine);
        //    if (iStreamInfoStartIndex >= 0)
        //    {
        //        string sDurationInfoLine = _colErrorLines[iStreamInfoStartIndex + 1];
        //        string sDurantionTime = sDurationInfoLine.Substring(12, 11);

        //        _tsDuration = VideoUtils.GetDurationFromFFMpegDurationString(sDurantionTime);
        //    }
        //}

        private void UpdateProgressFromOutputLine(string sOutputLine)
        {
            int iTimeIndex = sOutputLine.IndexOf("time=");
            int iBitrateIndex = sOutputLine.IndexOf(" bitrate=");

            string sCurrentTime = sOutputLine.Substring(iTimeIndex + 5, iBitrateIndex - iTimeIndex - 5);
            double nCurrentTimeInSeconds = double.Parse(sCurrentTime);
            double nPercentageComplete = nCurrentTimeInSeconds / _objSourceInfo.Duration.TotalSeconds;

            UpdateProgress(nPercentageComplete);
            //Console.WriteLine("Progress: " + _nPercentageComplete);
        }

        private void UpdateProgress(double nPercentageComplete)
        {
            _nPercentageComplete = nPercentageComplete;
            if (OnProgressEvent != null)
            {
                OnProgressEvent(this, new ProgressEventArgs( _nPercentageComplete));
            }
        }

        #endregion

        //public TimeSpan Duration { get { return _tsDuration; } }

        public double Progress { get { return _nPercentageComplete;  } }
        public string OutputFile { get { return _sOutputFile; } }

        public bool AudioEnabled
        {
            get { return _bAudioEnabled; }
            set { _bAudioEnabled = value; }
        }
}

public abstract class ConsoleCommandBase : CommandBase, ICommand
    {
        protected ProcessRunner _objCurrentProcessRunner;
        protected   List<String> _colOutputLines;
        protected List<String> _colErrorLines;


        private int _iExitCode;

        public ConsoleCommandBase()
        {
            _colOutputLines = new List<string>();
            _colErrorLines = new List<string>();
        }

        protected void InternalExecute(string sExePath, string sArgs)
        {
            InternalExecute(sExePath, sArgs, null, null, null);
        }

        protected void InternalExecute(string sExePath, string sArgs, string sDomain, string sUsername, string sPassword)
        {
            try
            {
                if (_objCurrentProcessRunner == null || _bIsRunning == false)
                {
                    StringReader objStringReader = new StringReader(string.Empty);

                    _objCurrentProcessRunner = new ProcessRunner(sExePath, sArgs);

                    _objCurrentProcessRunner.SetCredentials(sDomain, sUsername, sPassword);

                    _objCurrentProcessRunner.OutputReceived += new ProcessOutputEventHandler(OnOutputRecieved);
                    _objCurrentProcessRunner.ProcessExited += new ProcessExitedEventHandler(OnProcessExit);
                    _objCurrentProcessRunner.Run();

                    _bIsRunning = true;
                    _bIsComplete = false;
                }
                else
                {
                    DispatchException(new Exception("Processor Already Running"));
                }
            }
            catch (Exception objEx)
            {
                DispatchException(objEx);
            }
        }

        protected virtual void OnOutputRecieved(object sender, ProcessOutputEventArgs args)
        {
            try
            {
                if (args.Error == true)
                {
                    _colErrorLines.Add(args.Data);
                    //Console.WriteLine("Error: " + args.Data);
                }
                else
                {
                    _colOutputLines.Add(args.Data);
                    //Console.WriteLine(args.Data);
                }
            }
            catch (Exception objEx)
            {
                DispatchException(objEx);
            }
        }

        protected virtual void OnProcessExit(object sender, ProcessExitedEventArgs args)
        {
            try
            {
                Console.Write(ConsoleOutput);
                _iExitCode = args.ExitCode;

                _bIsRunning = false;
                _bIsComplete = true;

                //Some commands actually fail to succeed
                //if(args.ExitCode != 0)
                //{
                //    DispatchException(new Exception("Command Failed: " + this.GetType().Name + "\nConsole: " + ConsoleOutput + "\nConsoleError: " + ErrorOutput));
                //}

                OnInternalCommandComplete(_iExitCode);

                if (_objCurrentProcessRunner != null)
                {
                    _objCurrentProcessRunner.Dispose();
                    _objCurrentProcessRunner = null;    
                }
            }
            catch (Exception objEx)
            {
                DispatchException(objEx);
            }
        }

        abstract protected void OnInternalCommandComplete(int iExitCode);

        protected string JoinLines(List<String> colLines)
        {
            StringBuilder sbOutput = new StringBuilder();
            colLines.ForEach( o => sbOutput.AppendLine(o));
            return sbOutput.ToString();
        }

        #region Properties
        public int ExitCode
        {
            get { return _iExitCode; }
        }
        #endregion

        public override string GetCommandInfo()
        {
            StringBuilder sbCommandInfo = new StringBuilder();
            sbCommandInfo.AppendLine("Command:  " + this.GetType().Name);
            sbCommandInfo.AppendLine("Console Output");
            if (_colOutputLines != null)
            {
                foreach (string sOutputLine in _colOutputLines)
                {
                    sbCommandInfo.AppendLine("\t" + sOutputLine);
                }
            }
            sbCommandInfo.AppendLine("Error Output");
            if (_colErrorLines != null)
            {
                foreach (string sErrorLine in _colErrorLines)
                {
                    sbCommandInfo.AppendLine("\t" + sErrorLine);
                }
            }
            return sbCommandInfo.ToString();
        }

        public String ConsoleOutput { get { return JoinLines(_colOutputLines); } }
        public String ErrorOutput { get { return JoinLines(_colErrorLines);} }

    }

CommandBase : ICommand
    {
        protected IDedooseContext _context;
        protected Boolean _bIsRunning = false;
        protected Boolean _bIsComplete = false;

        #region Custom Events
        public event CommandCompleteEventHandler OnCommandComplete;
        event CommandCompleteEventHandler ICommand.OnCommandComplete
        {
            add { if (OnCommandComplete != null) { lock (OnCommandComplete) { OnCommandComplete += value; } } else { OnCommandComplete = new CommandCompleteEventHandler(value); } }
            remove { if (OnCommandComplete != null) { lock (OnCommandComplete) { OnCommandComplete -= value; } } }
        }

        public event UnhandledExceptionEventHandler OnCommandException;
        event UnhandledExceptionEventHandler ICommand.OnCommandException
        {
            add { if (OnCommandException != null) { lock (OnCommandException) { OnCommandException += value; } } else { OnCommandException = new UnhandledExceptionEventHandler(value); } }
            remove { if (OnCommandException != null) { lock (OnCommandException) { OnCommandException -= value; } } }
        }

        public event ProgressEventHandler OnProgressUpdate;
        event ProgressEventHandler ICommand.OnProgressUpdate
        {
            add { if (OnProgressUpdate != null) { lock (OnProgressUpdate) { OnProgressUpdate += value; } } else { OnProgressUpdate = new ProgressEventHandler(value); } }
            remove { if (OnProgressUpdate != null) { lock (OnProgressUpdate) { OnProgressUpdate -= value; } } }
        }
        #endregion

        protected CommandBase()
        {
            _context = UnityGlobalContainer.Instance.Context;
        }

        protected void DispatchCommandComplete(CommandResultType enResult)
        {
            if (enResult == CommandResultType.Fail)
            {
                StringBuilder sbMessage = new StringBuilder();
                sbMessage.AppendLine("Command Commpleted with Failure: "  + this.GetType().Name);
                sbMessage.Append(GetCommandInfo());
                Exception objEx = new Exception(sbMessage.ToString());
                DispatchException(objEx);
            }
            else
            {
                if (OnCommandComplete != null)
                {
                    OnCommandComplete(this, new CommandCompleteEventArgs(enResult));
                }
            }
        }

        protected void DispatchException(Exception objEx)
        {
            if (OnCommandException != null)
            { 
                OnCommandException(this, new UnhandledExceptionEventArgs(objEx, true)); 
            }
            else
            {
                _context.Logger.LogException(objEx, MethodBase.GetCurrentMethod());
                throw objEx;
            }
        }

        protected void DispatchProgressUpdate(double nProgressRatio)
        {
            if (OnProgressUpdate != null) { OnProgressUpdate(this, new ProgressEventArgs(nProgressRatio)); } 
        }

        public virtual string GetCommandInfo()
        {
            return "Not Implemented: " + this.GetType().Name;
        }

        public virtual void Execute() { throw new NotImplementedException(); }
        public virtual void Abort() { throw new NotImplementedException(); }

        public Boolean IsRunning { get { return _bIsRunning; } }
        public Boolean IsComplete { get { return _bIsComplete; } }

        public double GetProgressRatio()
        {
            throw new NotImplementedException();
        }
    }

public delegate void CommandCompleteEventHandler(object sender, CommandCompleteEventArgs e);

    public interface ICommand
    {
        event CommandCompleteEventHandler OnCommandComplete;
        event UnhandledExceptionEventHandler OnCommandException;
        event ProgressEventHandler OnProgressUpdate;

        double GetProgressRatio();
        string GetCommandInfo();

        void Execute();
        void Abort();
    }

//プロセスランナーについては、RogerKnappによるProcessRunnerを検索してください


1
        string result = String.Empty;
        StreamReader srOutput = null;
        var oInfo = new ProcessStartInfo(exePath, parameters)
        {
            UseShellExecute = false,
            CreateNoWindow = true,
            RedirectStandardOutput = true,
            RedirectStandardError = true
        };

        var output = string.Empty;

        try
        {
            Process process = System.Diagnostics.Process.Start(oInfo);
            output = process.StandardError.ReadToEnd();
            process.WaitForExit();
            process.Close();
        }
        catch (Exception)
        {
            output = string.Empty;
        }
        return output;

このラッパーは、メソッドをループに陥らせません。これを試してみてください、それは私のために働きました。


1

codeplexからFFPMEG.netをフォークしました。

まだ積極的に取り組んでいます。

https://github.com/spoiledtechie/FFMpeg.Net

dllを使用せず、exeを使用します。したがって、より安定する傾向があります。


私が求めているもののように見えますが、プロジェクトでこれをどのように実装しますか?
TEK 2016年

このプロジェクトをプロジェクトに追加し、FFMPEGがプロジェクト内に正しく配置されていることを確認します。そのまだ取り組んでいます。
SpoiledTechie.com 2016

このFFMPEG.netを使用してフレームをbyte []としてエンコードおよびデコードできますか?たとえば、byte [] encodeh264(byte [])およびbyte [] decodeh264(byte [])。
アフマド

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.