Microsoft MVP성태의 닷넷 이야기
.NET Framework: 914. C# - Task.Yield 사용법 [링크 복사], [링크+제목 복사],
조회: 12857
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 1개 있습니다.)
(시리즈 글이 4개 있습니다.)
.NET Framework: 914. C# - Task.Yield 사용법
; https://www.sysnet.pe.kr/2/0/12241

.NET Framework: 916. C# - Task.Yield 사용법 (2)
; https://www.sysnet.pe.kr/2/0/12245

.NET Framework: 1163.  C# - 윈도우 환경에서 usleep을 호출하는 방법
; https://www.sysnet.pe.kr/2/0/12980

.NET Framework: 1195. C# - Thread.Yield와 Thread.Sleep(0)의 차이점(?)
; https://www.sysnet.pe.kr/2/0/13033




C# - Task.Yield 사용법

Task.Yield가 기존의 스레드에서 제공하던 Yield와,

Thread.Yield 메서드
; https://learn.microsoft.com/ko-kr/dotnet/api/system.threading.thread.yield

유사하다고 생각해서 그동안 별 신경을 안 썼는데 우연히 아래의 글을 읽고,

When would I use Task.Yield()?
; https://stackoverflow.com/questions/22645024/when-would-i-use-task-yield

새롭게 봤습니다. ^^




제가 쓴 책을 보면 "10.2.4 Async 메서드가 아닌 경우의 비동기 처리" 절에서 "동기" 호출만 지원하는 메서드를 비동기로 제공하는 방법을 설명한 부분이 있습니다. 그게 Task.Yield의 기능을 설명하기 딱 좋은 사례일 듯합니다.

그러니까, File.ReadAllText와 같은 메서드의 경우 다음과 같이 동기 호출만 지원합니다.

static void OutputHOSTS()
{
    string text = File.ReadAllText(@"C:\Windows\system32\drivers\etc\HOSTS");
    WriteLog(text);
}

이 메서드를 비동기 호출로 변경하려면, 쉽게는 Task.Factory.StartNew의 힘을 빌려 다음과 같이 작성할 수 있습니다.

static async Task OutputHOSTSAsync()
{
    string text = await ReadAllTextAsync();
    WriteLog(text);
}

static Task<string> ReadAllTextAsync()
{
    return Task.Factory.StartNew(
        () => File.ReadAllText(@"C:\Windows\system32\drivers\etc\HOSTS")
        );
}

그런데, 이것보다 더 쉬운 방법이 바로 Task.Yield를 쓰는 것입니다.

static async Task OutputHOSTSAsync()
{
    string text = await ReadAllTextAsync2();
    WriteLog(text);
}

static async Task<string> ReadAllTextAsync2()
{
    await Task.Yield();
    return File.ReadAllText(@"C:\Windows\system32\drivers\etc\HOSTS");
}

어차피 중요한 것은, 스레드 풀의 스레드로 작업 실행을 넘기는 것이므로, 마이크로소프트는 그 역할을 Task.Yield 메서드에 구현해 두었고 우리는 저런 상황에서 사용해 주면 되는 것입니다.




하지만, Task.Yield를 이용한 비동기 처리 방법은 SynchronizationContext가 마련된 환경에서는 효과가 없습니다. 왜냐하면 그 메서드가 SynchronizationContext를 고려하기 때문인데, "When would I use Task.Yield()?" 글의 덧글을 보면 이에 대해 Task.Yield의 내부 동작은 다음과 동일하다는 설명이 나옵니다.

var tcs = new TaskCompletionSource<bool>();
var sc = SynchronizationContext.Current;
if (sc != null)
    sc.Post(_ => tcs.SetResult(true), null);
else
    ThreadPool.QueueUserWorkItem(_ => tcs.SetResult(true));
await tcs.Task;

따라서, Windows Forms/WPF 등의 UI 스레드 환경에서 Task.Yield를 호출하게 되면 이후의 코드는 UI 스레드에서 작업하게 되고, 대부분의 경우 그것은 Task.Yield의 호출 유무에 영향을 받지 않게 됩니다. 일례로, 해당 글의 덧글에서 Task.Yield가 SynchronizationContext를 인식함으로 차이가 있다는 예제 코드를 자세히 보면,

// 개인적으로, 해당 질문/답변 글의 문맥에서 봤을 때 이 예제는 잘못된 답변이라고 봅니다.
private async void button_Click(object sender, EventArgs e)
{
    await Task.Yield(); // Make us async right away

    var data = ExecuteFooOnUIThread(); // This will run on the UI thread at some point later

    await UseDataAsync();
}

Task UseDataAsync()
{
    return Task.Factory.StartNew(() => {
        Debug.WriteLine("TEST"); });
}

어차피 Task.Yield가 없어도 - button_Click 이벤트의 실행 자체가 UI 스레드에서 실행되므로 ExecuteFooOnUIThread 메서드에 대해 Task.Yield의 호출이 있을 필요가 없습니다. 반면, 다음과 같이 스레드 풀의 스레드가 await 이후의 코드를 실행하는 환경에서는,

private async void button_Click(object sender, EventArgs e)
{
    await UseDataAsync().ConfigureAwait(false); // 이후의 코드는 SynchronizationContext.Current == null인 스레드 풀의 스레드에서 실행되므로,

    await Task.Yield(); // Task.Yield 내부에서는 어차피 UI 스레드로 되돌릴 수도 없어,
    var data = ExecuteFooOnUIThread(); // 이 코드는 스레드 풀의 스레드에서 실행
}

Task.Yield는 이후의 코드를 UI 스레드에서 실행할 수 없으므로 마찬가지로 호출 여부가 크게 의미가 없습니다. 결국 SynchronizationContext.Current가 설정된 환경에서는 Task.Yield를 사용할 이유가 없는 것입니다. 실제로, 이 글의 처음 예제에서 만들어 둔 ReadAllTextAsync2 코드를 button_Click 이벤트에서 실행해 보면,

private async void button_Click(object sender, EventArgs e)
{
    string text = await ReadAllTextAsync2();
}

static async Task ReadAllTextAsync2()
{
    await Task.Yield();
    return File.ReadAllText(@"C:\Windows\system32\drivers\etc\HOSTS");
}

button_Click 및 ReadAllTextAsync2의 호출은 UI 스레드에서 진행되고, Task.Yield는 SynchronizationContext.Current를 고려하므로 마찬가지로 그 이후의 File.ReadAllText 역시 UI 스레드에서 진행되어 비동기 호출이라는 장점이 없어집니다.




개발자가 이런 사실을 적절히 잘 이해한다면, 특정 환경의 경우 Task.Yield는 기존 코드에서 스레드를 사용하던 부분에 대한 코드를 간단하게 만드는 용도라고 이해하는 것도 나쁘진 않습니다.

예를 들어 볼까요? 간혹 TCP 서버를 만들면서 접속한 클라이언트에 대해 하나의 스레드를 할당해 처리를 하는 경우가 있습니다.

using System;
using System.Net;
using System.Net.Sockets;
using System.Threading;

namespace TCPServer
{
    class Program
    {
        static void Main(string[] args)
        {
            using (Socket srvSocket = new Socket(AddressFamily.InterNetwork,
                 SocketType.Stream, ProtocolType.Tcp))
            {
                IPEndPoint endPoint = new IPEndPoint(IPAddress.Any, 11200);
                srvSocket.Bind(endPoint);
                srvSocket.Listen(10);

                while (true)
                {
                    Socket clntSocket = srvSocket.Accept();

                    Thread t = new Thread(processClientSocket);
                    t.IsBackground = true;
                    t.Start(clntSocket);
                }
            }
        }

        private static void processClientSocket(object objSock)
        {
            Socket clntSocket = objSock as Socket;

            // ... 다른 스레드에서 실행 ...
            Console.WriteLine("new client");

            clntSocket.Close();
        }
    }
}

이럴 때, Task.Yield를 사용해 기존 코드를 대체할 수 있습니다.

using System;
using System.Net;
using System.Net.Sockets;
using System.Threading;
using System.Threading.Tasks;

namespace TCPServer
{
    class Program
    {
        static async Task Main(string[] args)
        {
            using (Socket srvSocket = new Socket(AddressFamily.InterNetwork,
                 SocketType.Stream, ProtocolType.Tcp))
            {
                IPEndPoint endPoint = new IPEndPoint(IPAddress.Any, 11200);
                srvSocket.Bind(endPoint);
                srvSocket.Listen(10);

                while (true)
                {
                    Socket clntSocket = srvSocket.Accept();

                    /* await */ processClientSocketAsync(clntSocket);
                }
            }
        }

        static async Task processClientSocketAsync(Socket clntSocket)
        {
            await Task.Yield();

            // ... 다른 스레드에서 실행 ...
            Console.WriteLine("new client");

            clntSocket.Close();
        }
    }
}

그러니까, 일정 부분 이후의 기존 코드를 별도의 스레드에서 실행하기를 원한다면 Task.Yield로 간단하게 대체해 줄 수 있는 것입니다. (물론, SynchronizationContext가 구성된 환경에서는 단일 스레드로 동작하게 됩니다.)

이 정도면, Task.Yield에 대한 느낌이 대충 오시나요? ^^

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




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 12/7/2022]

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

비밀번호

댓글 작성자
 




... 61  62  [63]  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12072정성태12/9/201910931오류 유형: 583. iisreset 수행 시 "No such interface supported" 오류
12071정성태12/9/201913255오류 유형: 582. 리눅스 디스크 공간 부족 및 safemode 부팅 방법
12070정성태12/9/201915421오류 유형: 581. resize2fs: Bad magic number in super-block while trying to open /dev/.../root
12069정성태12/2/201911777디버깅 기술: 139. windbg - x64 덤프 분석 시 메서드의 인자 또는 로컬 변수의 값을 확인하는 방법
12068정성태11/28/201915126디버깅 기술: 138. windbg와 Win32 API로 알아보는 Windows Heap 정보 분석 [3]파일 다운로드2
12067정성태11/27/201911783디버깅 기술: 137. 실제 사례를 통해 Debug Diagnostics 도구가 생성한 닷넷 웹 응용 프로그램의 성능 장애 보고서 설명 [1]파일 다운로드1
12066정성태11/27/201911642디버깅 기술: 136. windbg - C# PInvoke 호출 시 마샬링을 담당하는 함수 분석 - OracleCommand.ExecuteReader에서 OpsSql.Prepare2 PInvoke 호출 분석
12065정성태11/25/201910510디버깅 기술: 135. windbg - C# PInvoke 호출 시 마샬링을 담당하는 함수 분석파일 다운로드1
12064정성태11/25/201912710오류 유형: 580. HTTP Error 500.0/500.33 - ANCM In-Process Handler Load Failure
12063정성태11/21/201911724디버깅 기술: 134. windbg - RtlReportCriticalFailure로부터 parameters 정보 찾는 방법
12062정성태11/21/201911814디버깅 기술: 133. windbg - CoTaskMemFree/FreeCoTaskMem에서 발생한 덤프 분석 사례 - 두 번째 이야기
12061정성태11/20/201911980Windows: 167. CoTaskMemAlloc/CoTaskMemFree과 윈도우 Heap의 관계
12060정성태11/20/201912352디버깅 기술: 132. windbg/Visual Studio - HeapFree x64의 동작 분석
12059정성태11/20/201911952디버깅 기술: 131. windbg/Visual Studio - HeapFree x86의 동작 분석
12058정성태11/19/201912749디버깅 기술: 130. windbg - CoTaskMemFree/FreeCoTaskMem에서 발생한 덤프 분석 사례
12057정성태11/18/20199882오류 유형: 579. Visual Studio - Memory 창에서 유효한 주소 영역임에도 "Unable to evaluate the expression." 오류 출력
12056정성태11/18/201913721개발 환경 구성: 464. "Microsoft Visual Studio Installer Projects" 프로젝트로 EXE 서명 및 MSI 파일 서명 방법파일 다운로드1
12055정성태11/17/20199449개발 환경 구성: 463. Visual Studio의 Ctrl + Alt + M, 1 (Memory 1) 등의 단축키가 동작하지 않는 경우
12054정성태11/15/201910763.NET Framework: 869. C# - 일부러 GC Heap을 깨뜨려 GC 수행 시 비정상 종료시키는 예제
12053정성태11/15/201912465Windows: 166. 윈도우 10 - 명령행 창(cmd.exe) 속성에 (DotumChe, GulimChe, GungsuhChe 등의) 한글 폰트가 없는 경우
12052정성태11/15/201911558오류 유형: 578. Azure - 일정(schedule)에 등록한 runbook이 1년 후 실행이 안 되는 문제(Reason - The key used is expired.)
12051정성태11/14/201914023개발 환경 구성: 462. 시작하자마자 비정상 종료하는 프로세스의 메모리 덤프 - procdump [1]
12050정성태11/14/201911697Windows: 165. AcLayers의 API 후킹과 FaultTolerantHeap
12049정성태11/13/201911795.NET Framework: 868. (닷넷 프로세스를 대상으로) 디버거 방식이 아닌 CLR Profiler를 이용해 procdump.exe 기능 구현
12048정성태11/12/201912564Windows: 164. GUID 이름의 볼륨에 해당하는 파티션을 찾는 방법
12047정성태11/12/201914409Windows: 163. 안전하게 eject시킨 USB 장치를 물리적인 재연결 없이 다시 인식시키는 방법
... 61  62  [63]  64  65  66  67  68  69  70  71  72  73  74  75  ...