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

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을 하게 되면 위에서 설명했던 것과 마찬가지로 풀링에 이미 반환돼 다른 목적으로 사용될 수 있으므로 오동작을 할 수 있는 것입니다.
정성태

... 31  32  33  34  35  36  37  38  39  40  41  42  [43]  44  45  ...
NoWriterDateCnt.TitleFile(s)
4803헬조선식...2/7/201721100WPF로 메신져 메세지 박스 구현 문의 [2]
4802spow...2/6/201718239특정 범위의 값을 다른 값으로 치환하는 메소드를 만들어야 할 때 가장 빠른 방법은? [7]
4804spow...2/7/201717443    답변글 [답변]: 특정 범위의 값을 다른 값으로 치환하는 메소드를 만들어야 할 때 가장 빠른 방법은?
4805spow...2/7/201717260    답변글 [답변]: 특정 범위의 값을 다른 값으로 치환하는 메소드를 만들어야 할 때 가장 빠른 방법은? (글삭제가 안돼 재첨부합니다)파일 다운로드1
4801spow...2/3/201717104C#의 참조 동작관련하여 아이디어 문의 드립니다 [4]
4800오세운2/1/201721370C# 에서 C++ DLL임포트 질문입니다. [2]
4798guest1/29/201719139picturebox의 image.dispose [1]
4797김철환1/13/201719099책에 관한 질문입니다 [3]
4796Bere...1/13/201718793++ 후위연산자와 = 을 함께 사용할 때 생성되는 IL 코드 관련... [2]
4795김철환1/11/201720431이벤트 부분을 읽고 있는데 이해가 안되서 질문합니다.. [11]
4794김철환1/10/201717403안녕하세요 c# 6.0 책을 구매한 사람인데요 [3]
4793장준영1/7/201719469안녕하세요 c언어 처음 공부해보는 학생입니다 [4]파일 다운로드1
4792김재영1/4/201721766소스코드 공개 전 성태님의 의견을 듣고싶습니다 [3]
4791C#초보12/28/201621312비동기 소켓 close시 ObjectDisposedException 문제점 질문 있습니다.. [1]
4790미나리12/24/201621496파워포인트 쇼 제어 SimpleHttpServer.cs 작동문제 [4]파일 다운로드1
4789김솔지12/21/201619485프린트 시, 프린트하는 파일의 파일명 구하는 부분에 대해서 질문드립니다. [1]
4788짜두12/19/201619932Visual Studio 2015 에서 msbuild 12 사용 [5]
4787guest12/18/201622448VLC라이브러리에 대해 아시나요? [3]파일 다운로드1
4785Hyou...12/16/201621599WPF 개발 시 MVVM 프레임워크 사용 [2]
4784ds12/15/201617638문의 드립니다. [2]
4783후배12/13/201618738MemoryStream에 관한 질문 입니다. [5]
4782김형민12/6/201617333[ C# 6.0 ] 126p 오타인가요? [6]
4781질문자11/29/201617318ms워드 저장 오류 [1]
4780최진11/28/201622004안녕 하세요 빌드 관련해서 질문드립니다 꾸벅 [4]
4779손니11/28/201618223안녕하세요 질문하다 드려도 될까요 [3]
4778김상호11/25/201617430재귀호출->비재귀호출 [2]파일 다운로드1
... 31  32  33  34  35  36  37  38  39  40  41  42  [43]  44  45  ...