C#使用FFMPEG推流,而且獲取流保存在本地,隨時取媒體進行播放!

   最近開發了基於C#的推流器一直不大理想,終於在不懈努力以後研究了一點成果,這邊作個筆記;本文着重在於講解下如何使用ffmpeg進行簡單的推流,看似簡單幾行代碼沒有官方的文檔很吃力。並獲取流的源代碼:以下→服務器

            #region RTMP推流(**已成功推流至服務器**)
            Network.Create()
                .WithSource(inputPath)//inputPath能夠改爲獲取設備的視頻流
                .WithDest("rtmp://192.168.61.128/live/livestream")//能夠根據本身的需求更新RTMP服務器地址
                .WithFilter(new X264Filter { ConstantQuantizer = 20 })
                .WithFilter(new ResizeFilter(Resolution.X720P))
                .Push();

            #endregion
            Network.Create()
                .WithSource("rtmp://192.168.61.128/live/livestream")//inputPath能夠改爲獲取設備的視頻流
                .WithDest(inputPath)//這個路徑能夠自由更改,若是是直播就不須要使用這個路徑,直接讀取流至播放器播放實時接收便可。
                .WithFilter(new X264Filter { ConstantQuantizer = 20 })
                .WithFilter(new ResizeFilter(Resolution.X720P))
                .Pull();

以上分別是推流和獲取流保存在本地的核心代碼。spa

1:首先創建服務器與客戶端的鏈接;3d

2:初始化服務器地址;code

3:初始化路徑;orm

4:視頻屬性設定;視頻

5:推/拉流操做;server

  /// <summary>
        /// 推流到RTMP服務器
        /// </summary>
        public void Push()
        {
            Validate();

            if (_destType != TargetType.Live)
            {
                throw new ApplicationException("當推流到RTMP服務器的時候,源類型必須是'RtmpType.Live'類型.");
            }

            //參數爲false的時候則爲推流
            var @params = GetParams(false);

            Processor.FFmpeg(@params);
        }
  /// <summary>
        /// 把流從RTMP服務器拉取--讀取視頻數據 ==pull a stream from rtmp server
        /// </summary>
        public void Pull()
        {
            Validate();

            if (!TestRtmpServer(_source, true))
                throw new ApplicationException("RTMP服務器發送錯誤.");

            if (_sourceType != TargetType.Live)
            {
                throw new ApplicationException("必須是RTMP服務器.");
            }
            //參數爲true的時候則爲讀取視頻流
            var @params = GetParams(false);

            Processor.FFmpeg(@params);
        }
 /// <summary>
        /// 檢測輸出輸入源以及過濾器
        /// </summary>
        private void Validate()
        {
            if (_sourceType == TargetType.Default)
                throw new ApplicationException("源錯誤.請輸入源!");

            if (_destType == TargetType.Default)
                throw new ApplicationException("dest錯誤.請輸入一個dest");

            var supportFilters = new[] { "Resize", "Segment", "X264", "AudioRate", "AudioBitrate" };

            if (_filters.Any(x => !supportFilters.Contains(x.Name)))
            {
                throw new ApplicationException(string.Format("過濾器不支持,過濾器只支持:{0} 類型",
                    supportFilters.Aggregate(string.Empty, (current, filter) => current + (filter + ",")).TrimEnd(new[] { ',' })));
            }
        }

這是推流所使用的方法,Validate()→這個方法主要用於:檢測輸出輸入源以及過濾器;其次檢測輸入流的狀態是否爲文件(File仍是Live);最後調用ffmpeg進行處理輸入的參數方法進行處理。blog

同理,獲取流也是如此。開發

接下來演示下效果:(推流成功以後就會有以下圖所示的效果,能夠自行調用Directshow等第三方播放器或者自帶的ffplay進行播放)文檔

 

 

 接下來是這個拉取流的效果:圖中未完成讀取的時候是下圖

值得注意的是,接收是從你開始接收的位置開始的,視頻推流是根據一幀一幀推送的,而咱們接收流的方式也是一幀一幀的接收,因此圖中顯示的實際上是已經播放到末尾的視頻,這樣防止了視頻重複上傳,重複下載。

 

備註:【思路僅供參考】

相關文章
相關標籤/搜索