欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

wpf實現(xiàn)超低延遲的RTMP或RTSP播放

 更新時間:2024年04月11日 08:24:17   作者:音視頻牛哥  
這篇文章主要為大家詳細介紹了wpf如何實現(xiàn)超低延遲的RTMP或RTSP播放,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下

?技術背景

我們在做Windows平臺RTMP和RTSP播放模塊對接的時候,有開發(fā)者需要在wpf下調(diào)用,如果要在wpf下使用,只需要參考C#的對接demo即可,唯一不同的是,視頻流數(shù)據(jù)顯示的話,要么通過控件模式,要么可以讓RTMP、RTSP播放模塊回調(diào)rgb數(shù)據(jù)上來,在wpf直接繪制即可。

技術實現(xiàn)

本文以大牛直播SDK的Windows平臺SmartPlayer為例,回調(diào)數(shù)據(jù)的模式,其他不再說明,只要處理好上來的數(shù)據(jù)就好:

播放之前,設置回調(diào),選擇NT_SP_E_VIDEO_FRAME_FORMAT_RGB32:

video_frame_call_back_ = new SP_SDKVideoFrameCallBack(SetVideoFrameCallBack);

NTSmartPlayerSDK.NT_SP_SetVideoFrameCallBack(player_handle_, (Int32)NT.NTSmartPlayerDefine.NT_SP_E_VIDEO_FRAME_FORMAT.NT_SP_E_VIDEO_FRAME_FORMAT_RGB32, IntPtr.Zero, video_frame_call_back_);

處理rgb數(shù)據(jù)回調(diào):

       /*
        * nt_player_wrapper.cs
        * Author: daniusdk.com
        */
        public void SetVideoFrameCallBack(IntPtr handle, IntPtr userData, UInt32 status, IntPtr frame)
        {
            if (frame == IntPtr.Zero)
            {
                return;
            }

            //如需直接處理RGB數(shù)據(jù),請參考以下流程
            NT_SP_VideoFrame video_frame = (NT_SP_VideoFrame)Marshal.PtrToStructure(frame, typeof(NT_SP_VideoFrame));

            if (video_frame.format_ != (Int32)NT.NTSmartPlayerDefine.NT_SP_E_VIDEO_FRAME_FORMAT.NT_SP_E_VIDEO_FRAME_FORMAT_RGB32)
                return;

            NT_SP_VideoFrame pVideoFrame = new NT_SP_VideoFrame();

            pVideoFrame.format_ = video_frame.format_;
            pVideoFrame.width_ = video_frame.width_;
            pVideoFrame.height_ = video_frame.height_;

            pVideoFrame.timestamp_ = video_frame.timestamp_;
            pVideoFrame.stride0_ = video_frame.stride0_;
            pVideoFrame.stride1_ = video_frame.stride1_;
            pVideoFrame.stride2_ = video_frame.stride2_;
            pVideoFrame.stride3_ = video_frame.stride3_;


            Int32 argb_size = video_frame.stride0_ * video_frame.height_;

            pVideoFrame.plane0_ = Marshal.AllocHGlobal(argb_size);
            CopyMemory(pVideoFrame.plane0_, video_frame.plane0_, (UInt32)argb_size);
        }

另外一種,可以用picturebox,在MainWindow.xaml 做以下設置:

        <WindowsFormsHost HorizontalAlignment="Left" Height="338" Margin="10,10,0,0" VerticalAlignment="Top" Width="480" Background="Black">
            <wf:PictureBox x:Name="RealPlayWnd"></wf:PictureBox>
        </WindowsFormsHost>

為了便于多實例集成參考,以播放2路為例(一路2560*1440,一路1920*1080):

具體實現(xiàn)如下:

       /*
        * MainWindow.xaml.cs
        * Author: daniusdk.com
        */
        public MainWindow()
        {
            InitializeComponent();

            if (!InitSDK())
                return;

            UIDispatcher = Dispatcher.CurrentDispatcher;

            player1_ = new nt_player_wrapper(RealPlayWnd, UIDispatcher);
            player1_.EventGetPlayerEventMsg += new DelGetPlayerEventMsg(GetPlayerEventMsgInfo);
            player1_.EventGetVideoSize += new DelGetVideoSize(GetVideoSize);

            player2_ = new nt_player_wrapper(RealPlayWnd1, UIDispatcher);
            player2_.EventGetPlayerEventMsg += new DelGetPlayerEventMsg(GetPlayerEventMsgInfo);
            player2_.EventGetVideoSize += new DelGetVideoSize(GetVideoSize);
        }

        private void GetPlayerEventMsgInfo(IntPtr handle, String msg)
        {
            this.Dispatcher.Invoke((Action)delegate()
            {
                event_label.Content = msg;
            });
        }

        private void GetVideoSize(IntPtr handle, String size)
        {
            this.Dispatcher.Invoke((Action)delegate()
            {
                video_size.Content = size;
            });
        }

        private bool InitSDK()
        {
            if (!is_player_sdk_init_)
            {
                UInt32 isPlayerInited = NT.NTSmartPlayerSDK.NT_SP_Init(0, IntPtr.Zero);
                if (isPlayerInited != 0)
                {
                    MessageBox.Show("調(diào)用NT_SP_Init失敗..");
                    return false;
                }

                is_player_sdk_init_ = true;
            }

            return true;
        }

        private void Button_Click_1(object sender, RoutedEventArgs e)
        {
            if (!player1_.IsPlaying())
            {
                player1_.SetBuffer(0);
                bool is_mute = true;

                if (!player1_.StartPlay("rtsp://admin:daniulive12345@192.168.0.120:554/h264/ch1/main/av_stream", false, is_mute))
                    return;

                btn_playback1.Content = "停止播放";
            }
            else
            {
                player1_.StopPlay();
                btn_playback1.Content = "開始播放";
            }
        }

        private void Button_Click_2(object sender, RoutedEventArgs e)
        {
            if (!player2_.IsPlaying())
            {
                player2_.SetBuffer(0);
                bool is_mute = true;

                if (!player2_.StartPlay("rtsp://admin:admin123456@192.168.0.121:554/cam/realmonitor?channel=1&subtype=0", false, is_mute))
                    return;

                btn_playback2.Content = "停止播放";
            }
            else
            {
                player2_.StopPlay();
                btn_playback2.Content = "開始播放";
            }
        }

        protected override void OnClosing(System.ComponentModel.CancelEventArgs e)
        {
            if (MessageBox.Show("確定要關閉窗口嗎?", "確認", MessageBoxButton.YesNo) != MessageBoxResult.Yes)
            {
                // 如果用戶選擇“否”,取消關閉
                e.Cancel = true;
            }

            if (player1_.IsPlaying())
            {
                player1_.StopPlay();
            }

            player1_.Dispose();

            if (player2_.IsPlaying())
            {
                player2_.StopPlay();
            }

            player2_.Dispose();

            if (is_player_sdk_init_)
            {
                NTSmartPlayerSDK.NT_SP_UnInit();
                is_player_sdk_init_ = false;
            }    

            base.OnClosing(e);
        }

延遲依舊毫秒級,CPU占用如下,如果用硬解碼,體驗會更好:

SmartPlayer以跨平臺的RTSP播放器為例,我們實現(xiàn)的功能如下,如不單獨說明,系Windows、Linux、Android、iOS全平臺支持:

  • [支持播放協(xié)議]高穩(wěn)定、超低延遲、業(yè)內(nèi)首屈一指的RTSP直播播放器SDK;
  • [多實例播放]支持多實例播放;
  • [事件回調(diào)]支持網(wǎng)絡狀態(tài)、buffer狀態(tài)等回調(diào);
  • [視頻格式]支持H.265、H.264,此外,還支持RTSP MJPEG播放;
  • [音頻格式]支持AAC/PCMA/PCMU;
  • [H.264/H.265軟解碼]支持H.264/H.265軟解;
  • [H.264硬解碼]Windows/Android/iOS支持特定機型H.264硬解;
  • [H.265硬解]Windows/Android/iOS支持特定機型H.265硬解;
  • [H.264/H.265硬解碼]Android支持設置Surface模式硬解和普通模式硬解碼;
  • [RTSP模式設置]支持RTSP TCP/UDP模式設置;
  • [RTSP TCP/UDP自動切換]支持RTSP TCP、UDP模式自動切換;
  • [RTSP超時設置]支持RTSP超時時間設置,單位:秒;
  • [RTSP 401認證處理]支持上報RTSP 401事件,如URL攜帶鑒權信息,會自動處理;
  • [緩沖時間設置]支持buffer time設置;
  • [首屏秒開]支持首屏秒開模式;
  • [復雜網(wǎng)絡處理]支持斷網(wǎng)重連等各種網(wǎng)絡環(huán)境自動適配;
  • [快速切換URL]支持播放過程中,快速切換其他URL,內(nèi)容切換更快;
  • [音視頻多種render機制]Android平臺,視頻:surfaceview/OpenGL ES,音頻:AudioTrack/OpenSL ES;
  • [實時靜音]支持播放過程中,實時靜音/取消靜音;
  • [實時音量調(diào)節(jié)]支持播放過程中實時調(diào)節(jié)音量;
  • [實時快照]支持播放過程中截取當前播放畫面;
  • [只播關鍵幀]Windows平臺支持實時設置是否只播放關鍵幀;
  • [渲染角度]支持0°,90°,180°和270°四個視頻畫面渲染角度設置;
  • [渲染鏡像]支持水平反轉(zhuǎn)、垂直反轉(zhuǎn)模式設置;
  • [等比例縮放]支持圖像等比例縮放繪制(Android設置surface模式硬解模式不支持);
  • [實時下載速度更新]支持當前下載速度實時回調(diào)(支持設置回調(diào)時間間隔);
  • [解碼前視頻數(shù)據(jù)回調(diào)]支持H.264/H.265數(shù)據(jù)回調(diào);
  • [解碼后視頻數(shù)據(jù)回調(diào)]支持解碼后YUV/RGB數(shù)據(jù)回調(diào);
  • [解碼前音頻數(shù)據(jù)回調(diào)]支持AAC/PCMA/PCMU數(shù)據(jù)回調(diào);
  • [音視頻自適應]支持播放過程中,音視頻信息改變后自適應;
  • [擴展錄像功能]完美支持和錄像模塊組合使用。

總結

Windows平臺下如果需要wpf播放,如果需要更靈活,可以采用回調(diào)rgb數(shù)據(jù)的模式,上層直接繪制,只是低延遲的播放出來畫面,采用上述控件模式亦可,除了wpf外,我們提供了C++和C#的接口和demo,感興趣的開發(fā)者,可以嘗試看看,有問題可以單獨跟我溝通。

到此這篇關于wpf實現(xiàn)超低延遲的RTMP或RTSP播放的文章就介紹到這了,更多相關wpf延遲播放內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • c#判斷代碼是否執(zhí)行超時的幾種方式總結

    c#判斷代碼是否執(zhí)行超時的幾種方式總結

    這篇文章主要介紹了c#判斷代碼是否執(zhí)行超時的幾種方式總結,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2023-01-01
  • 在WPF中使用Interaction.Triggers

    在WPF中使用Interaction.Triggers

    這篇文章介紹了在WPF中使用Interaction.Triggers的方法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-06-06
  • C#深度優(yōu)先搜索算法

    C#深度優(yōu)先搜索算法

    這篇文章主要介紹了C#深度優(yōu)先搜索算法,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-06-06
  • C#中幾個未知的Visual Studio編碼技巧分享

    C#中幾個未知的Visual Studio編碼技巧分享

    用了多年的Visual Studio,今天才發(fā)現(xiàn)這個編碼技巧,真是慚愧,分享出來,算是拋磚引玉吧,需要的朋友可以參考下
    2012-11-11
  • C# Invoke,begininvoke的用法詳解

    C# Invoke,begininvoke的用法詳解

    這篇文章主要介紹了C# Invoke,begininvoke的用法詳解,幫助大家更好的理解和使用c#,感興趣的朋友可以了解下
    2021-01-01
  • C#使用DLLImport調(diào)用外部DLL的方法

    C#使用DLLImport調(diào)用外部DLL的方法

    這篇文章介紹了C#使用DLLImport調(diào)用外部DLL的方法,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2022-04-04
  • C# menuStrip控件實現(xiàn)鼠標滑過自動彈出功能

    C# menuStrip控件實現(xiàn)鼠標滑過自動彈出功能

    MenuStrip 控件是 Visual Studio 和 .NET Framework 中的功能。使用該控件,可以輕松創(chuàng)建 Microsoft Office 中那樣的菜單。本文給大家分享menuStrip鼠標滑過自動彈出效果
    2021-07-07
  • 算法證明每一位都相同十進制數(shù)不是完全平方數(shù)

    算法證明每一位都相同十進制數(shù)不是完全平方數(shù)

    這篇文章主要為大家介紹了算法證明每一位都相同十進制數(shù)不是完全平方數(shù)的過程論述,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-05-05
  • C# Winform使用log4net進行日志記錄

    C# Winform使用log4net進行日志記錄

    Log4Net是從Java的log4j移植過來的,功能也與log4j類似,可以把日志信息輸出到文件、數(shù)據(jù)庫等不同的介質(zhì)或目標,下面我們就來學習一下如何使用log4net進行日志記錄吧
    2023-11-11
  • 詳解C#設置Excel數(shù)據(jù)自適應行高、列寬的2種情況

    詳解C#設置Excel數(shù)據(jù)自適應行高、列寬的2種情況

    這篇文章主要介紹了C#設置Excel數(shù)據(jù)自適應行高、列寬的2種情況,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-04-04

最新評論