C#에서 비동기 이벤트 처리: 비동기 작업을 안전하게 기다립니다
문제 개요:
C#에서 이벤트는 일반적으로 구독자에게 애플리케이션의 특정 이벤트를 알리는 데 사용되는 반환 값이 없는 대리자입니다. 그러나 비동기 이벤트 핸들러를 사용하려고 하면 문제가 발생할 수 있습니다. 즉, 이벤트가 발생하지만 비동기 핸들러 실행이 완료되기 전에 애플리케이션이 닫힙니다.
비동기 핸들러를 사용하여 이벤트 구독:
이벤트를 비동기적으로 처리하기 위해 이벤트 서명 자체를 수정하여 Task를 반환하는 것은 권장되지 않습니다. 대신 대기 가능한 콜백 함수를 허용하는 등록 메서드를 사용할 수 있습니다. 이를 통해 유연성이 향상되고 타사 구성 요소와 쉽게 통합할 수 있습니다.
등록 방식을 이용한 구현:
<code class="language-csharp">private List<Func<Task>> ShutdownCallbacks = new List<Func<Task>>(); public void RegisterShutdownCallback(Func<Task> callback) { this.ShutdownCallbacks.Add(callback); } public async Task Shutdown() { var callbackTasks = new List<Task>(); foreach (var callback in this.ShutdownCallbacks) { callbackTasks.Add(callback()); } await Task.WhenAll(callbackTasks); }</code>
이 예에서 ShutdownCallbacks
목록에는 기다릴 수 있는 콜백 함수가 저장되어 있습니다. Shutdown()
메서드는 이러한 콜백 함수를 반복하면서 동시에 기다립니다. 이렇게 하면 종료 프로세스가 계속되기 전에 모든 비동기 처리기가 완료됩니다.
통화 목록을 사용하여 이벤트 호출:
또는 기존 이벤트 패러다임을 유지하려면 GetInvocationList()
메서드를 사용하여 핸들러를 순차적으로 호출하고 반환된 작업을 기다릴 수 있습니다.
<code class="language-csharp">class A { public event Func<object, EventArgs, Task> Shutdown; public async Task OnShutdown() { Func<object, EventArgs, Task> handler = Shutdown; if (handler == null) { return; } Delegate[] invocationList = handler.GetInvocationList(); Task[] handlerTasks = new Task[invocationList.Length]; for (int i = 0; i < invocationList.Length; i++) { handlerTasks[i] = ((Func<object, EventArgs, Task>)invocationList[i])(this, EventArgs.Empty); } await Task.WhenAll(handlerTasks); } }</code>
여기서 OnShutdown()
는 이벤트 대리자를 가져오고 핸들러를 호출하며 완료될 때까지 기다립니다.
결론:
비동기 이벤트 핸들러가 매력적으로 보일 수 있지만 교착 상태 및 기타 문제가 발생할 가능성이 있으므로 일반적으로 이 접근 방식은 권장되지 않습니다. 위에서 언급한 것처럼 등록된 메서드나 호출 목록을 사용하면 C#에서 비동기적으로 기다리는 이벤트에 대한 솔루션을 더욱 강력하고 쉽게 관리할 수 있습니다.
위 내용은 C#에서 비동기 이벤트를 안전하게 기다리는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!