Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 7개 있습니다.)
.NET Framework: 394. async/await 사용 시 hang 문제가 발생하는 경우
; https://www.sysnet.pe.kr/2/0/1541

.NET Framework: 512. async/await 사용 시 hang 문제가 발생하는 경우 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/10801

.NET Framework: 631. async/await에 대한 "There Is No Thread" 글의 부가 설명
; https://www.sysnet.pe.kr/2/0/11129

.NET Framework: 720. 비동기 메서드 내에서 await 시 ConfigureAwait 호출 의미
; https://www.sysnet.pe.kr/2/0/11418

.NET Framework: 721. WebClient 타입의 ...Async 메서드 호출은 왜 await + 동기 호출 시 hang 현상이 발생할까요?
; https://www.sysnet.pe.kr/2/0/11419

디버깅 기술: 196. windbg - async/await 비동기인 경우 메모리 덤프 분석의 어려움
; https://www.sysnet.pe.kr/2/0/13563

닷넷: 2225. Windbg - dumasync로 분석하는 async/await 호출
; https://www.sysnet.pe.kr/2/0/13573




비동기 메서드 내에서 await 시 ConfigureAwait 호출 의미

async/await에 대한 내용(1, 2, 3)도 거의 정리가 되었으니 ^^ 이제 지난 글에서 소개한,

Async/Await - Best Practices in Asynchronous Programming
; https://learn.microsoft.com/en-us/archive/msdn-magazine/2013/march/async-await-best-practices-in-asynchronous-programming

async 메서드의 void 반환 타입 사용에 대하여
; https://www.sysnet.pe.kr/2/0/11414

다음의 의미를 파헤쳐 보겠습니다.

Use ConfigureAwait(false) when you can (예외인 경우: Methods that require con­text)

여기서 말하는 context란, SynchronizationContext를 의미합니다. 따라서 SynchronizationContext가 제공되지 않는 Console 응용 프로그램의 경우에는 ConfigureAwait 사용 유무에 따른 차이가 없습니다. 가령 다음의 프로그램은 ConfigureAwait에 true/false를 주는 것에 상관없이 출력이 동일합니다.

using System;
using System.Threading;
using System.Threading.Tasks;

namespace ConsoleApp1
{
    class Program
    {
        static bool captureContext = true;

        // C# 7.1 async Main
        static async Task Main(string[] args)
        {
            // 이 코드는 닷넷 프레임워크 환경에서 테스트한 것입니다. (참고: 닷넷 런타임에 따라 달라지는 AppDomain.GetCurrentThreadId의 반환값)
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " Main start");
            await AsyncMethod1().ConfigureAwait(captureContext);
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " Main end");
        }

        private static async Task AsyncMethod1()
        {
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " AsyncMethod1 start");
            await AsyncMethod2().ConfigureAwait(captureContext);
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " AsyncMethod1 end");
        }

        private static async Task AsyncMethod2()
        {
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " AsyncMethod2 start");
            await Task.Factory.StartNew(() => { Console.WriteLine("Task Thread: " + AppDomain.GetCurrentThreadId());   Thread.Sleep(5000); }).ConfigureAwait(captureContext);
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " AsyncMethod2 end");
        }
    }
}

/*
출력 결과:

20704 Main start
20704 AsyncMethod1 start
20704 AsyncMethod2 start
Task Thread: 27852
27852 AsyncMethod2 end
27852 AsyncMethod1 end
27852 Main end
*/

그런데, 정작 ConfigureAwait이 어떤 의미인지 말하지 않았군요. ^^ 이 메서드는 인자 값을 true로 실행하면 현재 스레드의 SynchronizationContext를 인지해 이후 실행해야 할 코드를 ThreadPool이 아닌, SynchronizationContext의 스레드에 태워 실행합니다. 따라서 이 차이점을 체험하려면 윈도우 폼/WPF/ASP.NET과 같은 응용 프로그램에서 테스트해야 합니다.

이를 위해 다음과 같이 간단하게 Windows Form로 코드 작성을 한 후,

using System;
using System.Diagnostics;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;

namespace WindowsFormsApp1
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        bool captureContext = true;

        private async void Form1_Load(object sender, EventArgs e)
        {
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " Main start");
            await AsyncMethod1().ConfigureAwait(captureContext);
            Trace.WriteLine(AppDomain.GetCurrentThreadId() + " Main end");
        }

        async Task AsyncMethod1()
        {
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " AsyncMethod1 start");
            await AsyncMethod2().ConfigureAwait(captureContext);
            Trace.WriteLine(AppDomain.GetCurrentThreadId() + " AsyncMethod1 end");
        }

        async Task AsyncMethod2()
        {
            Console.WriteLine(AppDomain.GetCurrentThreadId() + " AsyncMethod2 start");
            await Task.Factory.StartNew(() => { Trace.WriteLine("Task Thread: " + AppDomain.GetCurrentThreadId()); Thread.Sleep(5000); }).ConfigureAwait(captureContext);
            Trace.WriteLine(AppDomain.GetCurrentThreadId() + " AsyncMethod2 end");
        }
    }
}

실행하면, await 이후의 비동기 코드들이 모두 UI 스레드에서 실행된 것을 확인할 수 있습니다.

30040 Main start
30040 AsyncMethod1 start
30040 AsyncMethod2 start
Task Thread: 45800
30040 AsyncMethod2 end
30040 AsyncMethod1 end
30040 Main end

하지만 굳이 ConfigureAwait(true)를 지정할 필요는 없습니다. 왜냐하면 await의 기본 동작이 SynchronizationContext를 인지하도록 되어 있기 때문입니다.

반면, ConfigureAwait에 false를 설정하면 await 이후의 코드를 SynchronizationContext에 태우지 않고 Task.StartNew로 생성되었던 그 Task의 스레드를 이용해 실행하므로 결과가 다음과 같이 나옵니다. (콘솔 응용 프로그램에서의 동작과 같습니다.)

26892 Main start
26892 AsyncMethod1 start
26892 AsyncMethod2 start
Task Thread: 4176
4176 AsyncMethod2 end
4176 AsyncMethod1 end
4176 Main end

따라서, SynchronizationContext가 있는 상황에서 await 호출을 하는 경우, UI 객체를 건드리는 작업이 없다면 굳이 SynchronizationContext에 태울 필요가 없으므로 그런 경우에는 가능한 개발자가 ConfigureAwait(false)를 설정해 주는 것이 성능상 더 유리합니다. 이를 염두에 두고 "Async/Await - Best Practices in Asynchronous Programming" 글의 권고 사항이었던 다음의 문장을 다시 읽어 보면 의미가 파악될 것입니다.

Use ConfigureAwait(false) when you can (예외 조항: Methods that require con­text)

참고로, ConfigureAwait은 Task 타입과 TaskAwaiter 타입에서 구현된 기능이므로 엄밀히 async/await과는 독립적으로 바라봐야 합니다.

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




그런데, 이상하군요. 예전에 제가 쓴 글을 보면,

async/await 사용 시 hang 문제가 발생하는 경우
; https://www.sysnet.pe.kr/2/0/1541


ConfigureAwait을 false로 했는데도 hang 현상이 발생하는 경우가 있습니다. 그 이유는, 다음번 글에서 밝힙니다. ^^




(2024-04-10 업데이트)

ConfigureAwait(true) and ConfigureAwait(false)
; https://blog.stephencleary.com/2023/11/configureawait-in-net-8.html




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 4/10/2024]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 



2021-01-20 12시58분
[저예요~~] 글 잘봤습니다. (이제야... ^^;)
[guest]
2022-07-29 08시01분
[gwise] 한줄 정리하면 await가 UI를 건드리지 않는다면 ConfigureAwait(false) 이렇게 하는게 좋다." 감사합니다.
[guest]

... 106  107  108  109  110  111  112  113  114  115  116  117  118  [119]  120  ...
NoWriterDateCnt.TitleFile(s)
10949정성태4/28/201619834.NET Framework: 575. SharedDomain과 JIT 컴파일파일 다운로드1
10948정성태4/28/201623755.NET Framework: 574. .NET - 눈으로 확인하는 SharedDomain의 동작 방식 [3]파일 다운로드1
10947정성태4/27/201621626.NET Framework: 573. .NET CLR4 보안 모델 - 4. CLR4 보안 모델에서의 조건부 APTCA 역할파일 다운로드1
10946정성태4/26/201624472VS.NET IDE: 106. Visual Studio 2015 확장 - INI 파일을 위한 사용자 정의 포맷 기능 (Syntax Highlighting)파일 다운로드1
10945정성태4/26/201618197오류 유형: 327. VSIX 프로젝트 빌드 시 The "VsTemplatePaths" task could not be loaded from the assembly 오류 발생
10944정성태4/22/201619430디버깅 기술: 80. windbg - 풀 덤프 파일로부터 텍스트 파일의 내용을 찾는 방법
10943정성태4/22/201624278디버깅 기술: 79. windbg - 풀 덤프 파일로부터 .NET DLL을 추출/저장하는 방법 [1]
10942정성태4/19/201619623디버깅 기술: 78. windbg 사례 - .NET 예외가 발생한 시점의 오류 분석 [1]
10941정성태4/19/201619523오류 유형: 326. Error MSB8020 - The build tools for v120_xp (Platform Toolset = 'v120_xp') cannot be found.
10940정성태4/18/201622786Windows: 116. 프로세스 풀 덤프 시간을 줄여 주는 Process Reflection [3]
10939정성태4/18/201623812.NET Framework: 572. .NET APM 비동기 호출의 Begin...과 End... 조합 [3]파일 다운로드1
10938정성태4/13/201623387오류 유형: 325. 파일 삭제 시 오류 - Error 0x80070091: The directory is not empty.
10937정성태4/13/201631611Windows: 115. UEFI 모드로 윈도우 10 설치 가능한 USB 디스크 만드는 방법
10936정성태4/8/201642272Windows: 114. 삼성 센스 크로노스 7 노트북의 운영체제를 USB 디스크로 새로 설치하는 방법 [3]
10935정성태4/7/201626589웹: 32. Edge에서 Google Docs 문서 편집 시 한영 전환키가 동작 안하는 문제
10934정성태4/5/201625338디버깅 기술: 77. windbg의 콜스택 함수 인자를 쉽게 확인하는 방법 [1]
10933정성태4/5/201630939.NET Framework: 571. C# - 스레드 선호도(Thread Affinity) 지정하는 방법 [8]파일 다운로드1
10932정성태4/4/201623239VC++: 96. C/C++ 식 평가 - printf("%d %d %d\n", a, a++, a);
10931정성태3/31/201623498개발 환경 구성: 283. Hyper-V 내에 구성한 Active Directory 환경의 시간 구성 방법 [3]
10930정성태3/30/201621440.NET Framework: 570. .NET 4.5부터 추가된 CLR Profiler의 실행 시 Rejit 기능
10929정성태3/29/201631566.NET Framework: 569. ServicePointManager.DefaultConnectionLimit의 역할파일 다운로드1
10928정성태3/28/201637275.NET Framework: 568. ODP.NET의 완전한 닷넷 버전 Oracle ODP.NET, Managed Driver [2]파일 다운로드1
10927정성태3/25/201626511.NET Framework: 567. System.Net.ServicePointManager의 DefaultConnectionLimit 속성 설명
10926정성태3/24/201626030.NET Framework: 566. openssl의 PKCS#1 PEM 개인키 파일을 .NET RSACryptoServiceProvider에서 사용하는 방법 [10]파일 다운로드1
10925정성태3/24/201620352.NET Framework: 565. C# - Rabin-Miller 소수 생성 방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 - 두 번째 이야기파일 다운로드1
10924정성태3/22/201620959오류 유형: 324. Visual Studio에서 Azure 클라우드 서비스 생성 시 Failed to initialize the PowerShell host 에러 발생
... 106  107  108  109  110  111  112  113  114  115  116  117  118  [119]  120  ...