關於使用NAudio麥克風揚聲器組件造成WPF應用程式卡死問題跟蹤及非同步隊列的實現

来源:https://www.cnblogs.com/log9527blog/archive/2022/07/25/16517315.html
-Advertisement-
Play Games

由於WPF應用程式出現卡死的情況,特記錄一下問題的跟蹤情況 1、多次進行NAudio事件註冊,沒有啟用註銷再註冊的方式,造成應用程式CPU過高 private AudioNotificationClient audioNotification = new AudioNotificationClien ...


由於WPF應用程式出現卡死的情況,特記錄一下問題的跟蹤情況

1、多次進行NAudio事件註冊,沒有啟用註銷再註冊的方式,造成應用程式CPU過高

private AudioNotificationClient audioNotification = new AudioNotificationClient();

audioNotification.DeviceStateChanged += AudioNotification_DeviceStateChanged;

private MMDeviceEnumerator _mmDeviceEnumerator = new MMDeviceEnumerator();

_mmDeviceEnumerator.RegisterEndpointNotificationCallback(audioNotification);

缺少註銷

_mmDeviceEnumerator.UnregisterEndpointNotificationCallback(audioNotification);

2、事件註冊同時麥克風設備狀態發生改變DeviceStateChanged,造成線程死鎖

可以使用非同步隊列,把事件的註冊,註銷與DeviceStateChanged執行邏輯都放進非同步隊列,保證不會出現同時執行的情況。

非同步隊列的實現:

/// <summary>
    /// 非同步任務隊列
    /// </summary>
    public class AsyncTaskQueue : IDisposable
    {
        private bool _isDisposed;
        private readonly ConcurrentQueue<AwaitableTask> _queue = new ConcurrentQueue<AwaitableTask>();
        private Thread _thread;
        private AutoResetEvent _autoResetEvent;

        /// <summary>
        /// 非同步任務隊列
        /// </summary>
        public AsyncTaskQueue()
        {
            _autoResetEvent = new AutoResetEvent(false);
            _thread = new Thread(InternalRuning) {IsBackground = true};
            _thread.Start();
        }

        private bool TryGetNextTask(out AwaitableTask task)
        {
            task = null;
            while (_queue.Count > 0)
            {
                if (_queue.TryDequeue(out task) && (!AutoCancelPreviousTask || _queue.Count == 0)) return true;
                task.Cancel();
            }
            return false;
        }

        private AwaitableTask PenddingTask(AwaitableTask task)
        {
            lock (_queue)
            {
                Debug.Assert(task != null);
                _queue.Enqueue(task);
                _autoResetEvent.Set();
            }
            return task;
        }

        private void InternalRuning()
        {
            while (!_isDisposed)
            {
                if (_queue.Count == 0)
                {
                    _autoResetEvent.WaitOne();
                }
                while (TryGetNextTask(out var task))
                {
                    if (task.IsCancel) continue;

                    if (UseSingleThread)
                    {
                        task.RunSynchronously();
                    }
                    else
                    {
                        task.Start();
                    }
                }
            }
        }

        /// <summary>
        /// 是否使用單線程完成任務.
        /// </summary>
        public bool UseSingleThread { get; set; } = true;

        /// <summary>
        /// 自動取消以前的任務。
        /// </summary>
        public bool AutoCancelPreviousTask { get; set; } = false;

        /// <summary>
        /// 執行任務
        /// </summary>
        /// <param name="action"></param>
        /// <returns></returns>
        public AwaitableTask Run(Action action)
            => PenddingTask(new AwaitableTask(new Task(action, new CancellationToken(false))));

        /// <summary>
        /// 執行任務
        /// </summary>
        /// <typeparam name="TResult"></typeparam>
        /// <param name="function"></param>
        /// <returns></returns>
        public AwaitableTask<TResult> Run<TResult>(Func<TResult> function)
            => (AwaitableTask<TResult>) PenddingTask(new AwaitableTask<TResult>(new Task<TResult>(function)));


        /// <inheritdoc />
        public void Dispose()
        {
            Dispose(true);
            GC.SuppressFinalize(this);
        }
        
        /// <summary>
        /// 析構任務隊列
        /// </summary>
        ~AsyncTaskQueue() => Dispose(false);

        private void Dispose(bool disposing)
        {
            if (_isDisposed) return;
            if (disposing)
            {
                _autoResetEvent.Dispose();
            }
            _thread = null;
            _autoResetEvent = null;
            _isDisposed = true;
        }

        /// <summary>
        /// 可等待的任務
        /// </summary>
        public class AwaitableTask
        {
            private readonly Task _task;

            /// <summary>
            /// 初始化可等待的任務。
            /// </summary>
            /// <param name="task"></param>
            public AwaitableTask(Task task) => _task = task;

            /// <summary>
            /// 任務的Id
            /// </summary>
            public int TaskId => _task.Id;

            /// <summary>
            /// 任務是否取消
            /// </summary>
            public bool IsCancel { get; private set; }

            /// <summary>
            /// 開始任務
            /// </summary>
            public void Start() => _task.Start();

            /// <summary>
            /// 同步執行開始任務
            /// </summary>
            public void RunSynchronously() => _task.RunSynchronously();

            /// <summary>
            /// 取消任務
            /// </summary>
            public void Cancel() => IsCancel = true;

            /// <summary>
            /// 獲取任務等待器
            /// </summary>
            /// <returns></returns>
            public TaskAwaiter GetAwaiter() => new TaskAwaiter(this);

            /// <summary>Provides an object that waits for the completion of an asynchronous task. </summary>
            [HostProtection(SecurityAction.LinkDemand, ExternalThreading = true, Synchronization = true)]
            public struct TaskAwaiter : INotifyCompletion
            {
                private readonly AwaitableTask _task;

                /// <summary>
                /// 任務等待器
                /// </summary>
                /// <param name="awaitableTask"></param>
                public TaskAwaiter(AwaitableTask awaitableTask) => _task = awaitableTask;

                /// <summary>
                /// 任務是否完成.
                /// </summary>
                public bool IsCompleted => _task._task.IsCompleted;

                /// <inheritdoc />
                public void OnCompleted(Action continuation)
                {
                    var This = this;
                    _task._task.ContinueWith(t =>
                    {
                        if (!This._task.IsCancel) continuation?.Invoke();
                    });
                }
                /// <summary>
                /// 獲取任務結果
                /// </summary>
                public void GetResult() => _task._task.Wait();
            }
        }

        /// <summary>
        /// 可等待的任務
        /// </summary>
        /// <typeparam name="TResult"></typeparam>
        public class AwaitableTask<TResult> : AwaitableTask
        {
            /// <summary>
            /// 初始化可等待的任務
            /// </summary>
            /// <param name="task">需要執行的任務</param>
            public AwaitableTask(Task<TResult> task) : base(task) => _task = task;


            private readonly Task<TResult> _task;

            /// <summary>
            /// 獲取任務等待器
            /// </summary>
            /// <returns></returns>
            public new TaskAwaiter GetAwaiter() => new TaskAwaiter(this);

            /// <summary>
            /// 任務等待器
            /// </summary>
            [HostProtection(SecurityAction.LinkDemand, ExternalThreading = true, Synchronization = true)]
            public new struct TaskAwaiter : INotifyCompletion
            {
                private readonly AwaitableTask<TResult> _task;

                /// <summary>
                /// 初始化任務等待器
                /// </summary>
                /// <param name="awaitableTask"></param>
                public TaskAwaiter(AwaitableTask<TResult> awaitableTask) => _task = awaitableTask;

                /// <summary>
                /// 任務是否已完成。
                /// </summary>
                public bool IsCompleted => _task._task.IsCompleted;

                /// <inheritdoc />
                public void OnCompleted(Action continuation)
                {
                    var This = this;
                    _task._task.ContinueWith(t =>
                    {
                        if (!This._task.IsCancel) continuation?.Invoke();
                    });
                }

                /// <summary>
                /// 獲取任務結果。
                /// </summary>
                /// <returns></returns>
                public TResult GetResult() => _task._task.Result;
            }
        }

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 派生類繼承了基類除構造函數和析構函數外的所有數據成員和函數成員。派生類和基類存在一種特殊關係:派生類是一種基類,具有基類的所有功能。面向對象的程式設計利用派生類和基類之間的特殊關係,常常將派生類對象當作基類對象使用,或者用基類來代表派生類,其目的是提高代碼可重用性。由於C++對數據類型一致性要求比較 ...
  • Spring框架是什麼? Spring 是於 2003 年興起的一個輕量級的 Java 開發框架,它是為瞭解決企業應用開發的複雜性而創建的。Spring 的核心是控制反轉(IoC)和麵向切麵編程(AOP)。Spring 是可以在 Java SE/EE 中使用的輕量級開源框架。 Spring 的主要作 ...
  • 1.橋接方法簡介 橋接方法是jdk1.5引入泛型後,為使java泛型方法生成的位元組碼與jdk1.5版本之前的位元組碼相容由編譯器自動生成的。 可用method.isBridge() 判斷method是否是橋接方法,在生成的位元組碼中會有flags標記 ACC_BRIDGE, ACC_SYNTHETIC ...
  • 利用Python實現壓縮一個文件夾 二、知識點 文件讀寫 基礎語法 字元串處理 迴圈遍歷 文件壓縮 三、代碼解析 導入系統包 import platform import os import zipfile # 我還給大家準備了這些資料:Python視頻教程、100本Python電子書、基礎、爬蟲、 ...
  • 在上兩篇文章中已經將播放視頻的功能實現了,今天我就來講解一下如何通過FFmpeg來解析音頻內容,並且用NAudio來進行音頻播放; 效果圖 雖然效果圖是gif並不能 聽到音頻播放的內容,不過可以從圖中看到已經是實現了音頻的播放,暫停,停止已經更改進度的內容了; 一。添加NAudio庫: 一.音頻解碼 ...
  • 一.網路協議 如果要理解Socket,要熟悉TCP/IP即傳輸控制協議/網間協議,定義了主機如何連入網際網路,數據如何在它們之間傳輸的標準。 TCP/IP協議參考模型,把所有的TCP/IP系列協議歸類到四個抽象層中:應用層,傳輸層,網路層,數據鏈路層,每一抽象層建立在低一層提供的服務上,並且為高一層提 ...
  • 我自己用這些代碼做的小app如下: 第一種,user32.dll /// <summary> /// 調用外部切換壁紙的方法 /// </summary> /// <param name="uAction"></param> /// <param name="uParam"></param> /// ...
  • 製作一個用戶頭像選擇器仿 WeGame 製作一個用戶頭像選擇Canvas為父控制項所實現,展示圖片使用Image,Path當作上方的蒙版;Canvas:主要用途方便移動Image,設置ClipToBounds="True"裁剪為一個正方形200x200做為主要展示區域;Image:展示需要裁剪的圖片; ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...