Microsoft MVP성태의 닷넷 이야기
ValueTask를 multiple await 하지 말라는데 [링크 복사], [링크+제목 복사],
조회: 2573
글쓴 사람
eric (gss9282 at gmail.com)
홈페이지
첨부 파일
 
(연관된 글이 2개 있습니다.)

https://devblogs.microsoft.com/dotnet/understanding-the-whys-whats-and-whens-of-valuetask/

이 게시글이나 msdn 보면 ValueTask에 대해서 multiple await가 안좋다 하는데


제가 다음과 같이

var t = GetValue();
await t;
await t;

private async ValueTask<int> GetValue()
{
    if (cachedValue is null)
    {
        await Task.Delay(500);
        cachedValue = 5;
    }
    return cachedValue.Value;
}

이러면 CA2012 경고가 나오긴 하는데
작동은 잘됩니다.

어떠한 케이스에서 multiple await를 쓰면 예외가 터지는건가요?


[연관 글]






[최초 등록일: ]
[최종 수정일: 6/10/2025]


비밀번호

댓글 작성자
 



2025-06-11 04시00분
해당 글을 잠시 읽어보니까, 그 글에 답이 잘 나오는군요. ^^

multiple await을 할 때 예외가 발생할 수 있는 경우는 ValueTask가 IValueTaskSource를 구현한 개체를 감싸서 반환한 경우입니다. 예제로 든 경우는 TResult를 반환한 값이기 때문에 경고는 나와도 문제는 없습니다.

참고로, IValueTaskSource를 구현한 개체를 반환하더라도 문제가 안 되는 경우도 있습니다. 문제가 되는 경우는, 해당 문서에서도 예를 들어 나오지만 Pool에 관리가 된 경우입니다. 가령 예제에서 GetValue가 풀링된 개체를 담은 ValueTask를 반환했다고 가정하면 두 번째 await 호출 시에 이미 해당 개체는 풀에 반환돼 다른 값을 가지고 있거나 반환되면서 값이 초기화돼 정상적인 값을 구할 수 없게 될 것입니다.

게다가, IValueTaskSource를 감싼 경우라면 await을 분리해서 호출하는 경우에도 문제가 발생할 수 있습니다. 즉, 다음과 같이 직접 호출하는 식으로 바꿔야 합니다.

var t = await GetValue(); // 이렇게 메서드 호출에 직접 await 사용

왜냐하면, IValueTaskSource가 반환하면서 직접적인 값을 ValueTask<TResult>의 TResult를 '값 복사'해서 반환해 사용해야 안전한 것입니다. 그렇지 않고 ValueTask<TResult>로 받고 이후에 await을 하게 되면 위에서 설명했던 것과 마찬가지로 풀링에 이미 반환돼 다른 목적으로 사용될 수 있으므로 오동작을 할 수 있는 것입니다.

-----------------

아래의 글에 정리했으니 참고하세요.

C# - IValueTaskSource로 인해 주의가 필요한 ValueTask 호출
; https://www.sysnet.pe.kr/2/0/13952
정성태

1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
5790집으로 ...1/13/202311492[UI 멈춤 현상]deadlock 관련 글을 보고 혹시나 하고 문의 드립니다. [4]
5789guest1/10/202311110스레드와 Async Task [2]파일 다운로드1
5788kr11/10/202311512C# 에서 제공하는 컬렉션들의 차이점이 궁금합니다. [3]
5787stack1/10/202311552STACKOVERFLOW [1]
5786Dev ...1/9/202313551익명 클래스 말고 익명 구조체는 불가능한걸까요? [4]
5785음성인식1/8/202313391음성인식 System.Speech - 문법에 사용된 언어가 음성 인식기의 언어와 일치하지 않습니다. [2]
5784MS워드1/8/202312299MS워드에서 ctrl Z는 클립보드를 이용하나요? 아니면 참조자 이용하나요? [7]
5783구직자1/7/202311404C#개발자 구인광고와 초급개발자 [3]파일 다운로드1
5782Sqli...1/5/202312380윈도우11 노트북에서 exe(Sqlite)만들어 윈도우 7 PC에 설치 시 [5]
5781List맨1/5/202312367List.Add("newobj") 속도는 빠른 편인지요? [11]
5780임세1/3/202314467C# 프로그래밍 10 책 구매한 사람입니다. 3부 자료는 어디서 다운 받을 수 있나요? [4]
5779이건우1/2/202312576안녕하십니까 루프안에서 메세지처리에 관하여 질문드립니다! [2]
5778이건우12/30/202213270안녕하세요 c#에서 dll참조 관련 질문드립니다. [4]
5777감사합니...12/29/202212043UI Thread에 Invoke 처리관련 궁금합니다. [3]
5776pdf맨12/28/202211667C# MouseUp Event + pdf [10]
5775민성12/28/202211886안녕하세요 Class 관련해서 예외처리를 하나로 받아낼수 있는 방법 [1]
5774중급12/27/202213513중급개발자란 어느 수준인지요? [4]
5773김영식12/26/202212238c# 압축파일 읽어 올 때 BinaryRead 한글 처리 문제 [1]
5772눈송이12/26/202212678Excel VSTO 는 왜 Net Core, Net 5, 6 버전을 사용하지 않나요? [2]
5771김훈12/26/202212275c# .net client application 망분리(내부망,외부망) 환경에서 의문의 외부사이트 호출 대기 [2]
5770lsh12/26/202211773클라우디움안에 있는 파일을 File.Copy 하고싶은데 코드로는 접근을 못하나요? [1]
5769울타리12/20/202212480Active Directory 2012R2 2016 또는 2019 마이그렝션 문의 드립니다. [1]
5768c++12/14/202214928Thread를 사용한 C++ DLL에 관련된 질문입니다. [6]파일 다운로드2
5767민성12/9/202211950안녕하세요 ashx로 화일을 저장하고 화일명을 리턴하는데요 [1]
5766김명훈12/9/202212125웹브라우저에서 묻지 않고 바로 다운로드 [2]
5765hong12/1/202212560Winform(.Net6) 클라이언트에서 SignalR Core 웹서버에 접속시 인증서 문제 [3]파일 다운로드1
1  2  3  4  5  6  7  [8]  9  10  11  12  13  14  15  ...