Microsoft MVP성태의 닷넷 이야기
비동기 코드 흐름 질문있습니다. [링크 복사], [링크+제목 복사],
조회: 7910
글쓴 사람
한예ㅈ
홈페이지
첨부 파일
 

선생님 안녕하세요!
비동기 코드 흐름이 궁금해서 질문 드립니다!

example.txt 내용은 아래와 같습니다.
01.여기는 서울입니다.
02.여기는 경기도입니다.
03.여기는 부산입니다.

namespace Test
{
    class Program
    {
        static void Main(string[] args)
        {
            RunAsync();
            Console.ReadLine();
        }

        private static async void RunAsync()
        {
            string text = await TextReaderSample.ReadTextAsync("example.txt");
            Console.Write(text);
        }
    }

    static class TextReaderSample
    {
        public static async Task<string> ReadTextAsync(string filePath)
        {
            StringBuilder sb = new StringBuilder();
            StreamReader sr = new StreamReader(filePath);
            while (!sr.EndOfStream)
            {
                string line = await sr.ReadLineAsync(); // ★★★
                Print();
                sb.AppendLine(line);
            }
            return sb.ToString();
        }
        
        public static void Print()
        {
            Console.WriteLine("Main Thread Working");
        }
    }
}
[출력]
Main Thread Working
01.여기는 서울입니다.
02.여기는 경기도입니다.
03.여기는 부산입니다.

[질문 1] 저는 코드가 아래와 같이 동작한다고 생각하는데 혹시 틀린 부분이 있을까요?
① Main Thread가 RunAsync()를 실행(호출)한다.
② Main Thread가 TextReaderSample.ReadTextAsync("example.txt")를 실행(호출)한다.
③ Main Thread는 StringBuilder, StreamReader 객체를 생성 후
  반복문 안으로 들어온다.
④ ReadLineAsync는 스레드 풀에 있는 작업 스레드로 실행한다.
  작업 스레드가 ReadLineAsync 처리를 완료할 때까지
  Main Thread는 sr 앞에 있는 await 키워드에서 대기한다.
  작업 스레드가 ReadLineAsync 처리를 완료했다면 결과를 Main Thread가 넘겨받는다.
⑤ Main Thread는 Print()를 실행 후,
  StringBuilder 객체에 추가(AppendLine)한다.
⑤ Main Thread가 TextReaderSample.ReadTextAsync() 밖으로 벗어나서
  RunAsync()안에 있는 Console.Write(text);를 실행한다.
⑥ Main Thread가 RunAsync() 밖으로 벗어나서 Main 문으로 돌아온다.

[질문 2] ReadLineAsync()를 작업 스레드가 실행하는데
혹시 작업 스레드도 Main Thread처럼 ID 출력할 수 있는 방법이 있을까요?
선생님께서 알려주신 Thread.CurrentThread.ManagedThreadId로는 작업 스레드의 id를 확인하는 것이 어려운 것 같습니다..

[질문 3]
② string line = await sr.ReadLineAsync(); 를 아래와 같이 수정했다면

Task<string> returnedTaskTResult = sr.ReadLineAsync();
Print();
string line = await returnedTaskTResult;

스레드 풀에 있는 작업 스레드가 ReadLineAsync()를 처리하는 동안
Main Thread는 Print()를 처리한 후에 returnedTaskTResult 앞에 있는 await 키워드 앞에서
ReadLineAsync()의 결과 값을 기다린다고 생각해도 될까요?








[최초 등록일: ]
[최종 수정일: 5/23/2021]


비밀번호

댓글 작성자
 



2021-05-24 09시56분
[시린이] 안녕하세요. 저도 배우는 입장으로 제가 생각하는 것이 맞는지 답변 달아 봅니다.
잘못된 내용이 있다면 정성태님이 바로 잡아주실 것 같습니다.

[질문1]
다음과 같이 처리 되는 것으로 알고 있습니다.
1. 메인 스레드 Main() 진입
2. 메인스레드에서 RunAsync() 진입
    → 2-1. 메인 스레드에서 TextReaderSample.ReadTextAsync의 Task<string>를 생성해서 반환됨.
    → 2-2. 곧장 Console.Write(text) 실행
3. 메인 스레드에서 TextReaderSample.ReadTextAsync() 메서드 진입
    → 3-1. await sr.ReadLineAsync() 구문을 만나면 Task<string>를 반환하고 바로 return
    → 3-2. 스레드풀에서 작업 스레드가 ReadLineAsync()처리를 하고 sb.ToString() 반환까지 모두 처리

[질문2]
Thread.CurrentThread.ManagedThreadId 확인인 어려운 이유가 무엇인지 궁금합니다.

[질문3]
string line = await sr.ReadLineAsync();
구문을 Task변수를 선언해서 풀어서 기술한것이

Task<string> returnedTaskTResult = sr.ReadLineAsync();
string line = await returnedTaskTResult;
위 코드가 아닌가요? 즉, 같은 방식의 처리 아닌가요?
[guest]
2021-05-24 10시00분
자꾸 파고드시니... 이제 쉬운 답변은 할 수가 없군요, ^^; 지난번의 질문과 함께 포함한 답이라고 보면 됩니다.

답변 1) "4. ReadLineAsync는 스레드 풀에 있는 작업 스레드로 실행한다." 단계에 대한 것은 잘못된 이해입니다. 엄밀히 말해서는, 해당 비동기 작업은 (I/O를 포함한다면 내부의 device driver가 관여하는) 나름의 방법으로 정의된 절차로 진행됩니다. 여기서 스레드풀이 관여하는 곳은 ReadLineAsync 이후의 코드를 누가 수행할 것이냐하는 부분부터입니다. 이에 대해서는 다음의 글에서 좀 더 설명했으니 참고하세요.

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

그리고, 이후의 Main Thread가 sr 앞에 있는 await 키워드에서 대기한다는 것도 잘못되었습니다.

await 호출을 개념적으로 보면 "signal"을 전송한다고 이해를 하는 것이 편합니다. 스레드가 실행 중 await 호출을 만나면 - 위의 경우처럼 await sr.ReadLineAsync()" 호출을 만나면 그냥 read하라고 signal을 전송하고 지나가는 것입니다. 그렇다고 해서 그 아래의 Print를 호출하는 것은 아닙니다. 왜냐하면, 이후의 코드는 모두 분리돼 나중에 ReadLineAsync가 수행되고 난 후에 실행될 것이기 때문입니다. 이에 대해서는 제 책의 "10.2 비동기 호출" 절을 보시면 이해가 더 잘 될 것입니다.

답변 2) 답변 1에서 이미 답변한데로 그 이후의 작업은 device의 인터럽트 호출로 인한 비동기이기 때문에 스레드 id 확인 자체가 의미가 없습니다. 심지어 await 대상이 되는 코드가 I/O와 관련된 것이 아닌 경우 별도 스레드를 사용하고/하지 않고는 해당 Async 메서드를 개발하는 측의 자유입니다. 이에 대한 것도 다음의 글을 보시면 됩니다.

C# - 동기 방식이면서 비동기 메서드처럼 구현한 사례
; https://www.sysnet.pe.kr/2/0/11431

답변 3) 위의 내용으로 답변이 되었습니다.

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

async/await을 근본적으로 이해하고 싶다면 다음의 글을 참고하시고,

C# 컴파일러 대신 직접 구현하는 비동기(async/await) 코드
; https://www.sysnet.pe.kr/2/0/11351

마지막으로, 공부하는 입장에서는 이해를 위해 그렇게 예제를 만들 수 있지만, Async 메서드는 가능한 await으로 쓰는 것이 좋습니다. 이에 대해서도 다음의 글을 참고하세요.

Async/Await - Best Practices in Asynchronous Programming
  - async-all-the-way
; https://docs.microsoft.com/en-us/archive/msdn-magazine/2013/march/async-await-best-practices-in-asynchronous-programming#async-all-the-way
정성태
2021-05-24 10시30분
[한예지] 답변 감사드립니다! 교재 683쪽부터 읽어보고 다시 답변해주신 것에 대해 생각해보도록 할겠습니다!!
[guest]

... 46  47  48  49  [50]  51  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
3578이성환7/9/201513283WPF Multi Dispatcher 사용 시 hang 발생 [4]파일 다운로드1
3577초보개발자7/9/201512219C# SHDocVw.InternetExplorer 관련 도움좀 부탁드리겠습니다. [1]
3576솔솔7/6/201511032zip압축시! [2]
3575염기돈6/29/201511903책을 보다가 익명 메서드 관련해서 질문거리가 생겼습니다. [2]파일 다운로드1
3574김기술6/26/201510068프로그램 사용중 USB 경로 관련 해서 문의 드립니다 [4]
3573gagl...6/24/201512839IIS 의 특정 페이지들에서 20초의 딜레이가 있습니다. [2]
3572프란치스코6/24/201511307책을 구입하고 싶은데, [5]
3571로손6/22/201511368.Net COM dll 관련 사용시 Event 부분에 관한 문의 입니다. [5]파일 다운로드1
3570나그네6/11/201514200안녕하세요.. ^^ 궁금한게 있어서 혹시 조언을 얻을수 있을까 해서.. [2]
3569김지용6/9/201517114FFmpeg.exe 를 이용한 C# 동영상 인코더 예제보고 질문 드립니다. [9]
3568유동근6/5/201511582C# TTS 오류 입니다 도와주세요.파일 다운로드1
3567김보경5/26/201512063c#이용한 음성인식에 질문드립니다. [1]
3566로손5/26/201514417Http 파일 업로드시 한글파일명 관련 [1]
3565김태훈5/20/201510504AxWebBrowser 파일 다운로드 관련하여 문의드립니다. [3]
3564이강산5/20/201510363MD5 인코딩 관련 문의 입니다 [1]
2564CatO...5/13/201511132AppDomain 문제로 삽질중입니다. [2]
2563솔솔5/13/201510137custom search!! [1]
2562안녕하세요5/13/201510681안녕하세요.. C# 구조체 관련. [1]
2560솔솔5/6/201510623zip파일생성시. [1]
2561솔솔5/6/201511095    답변글 [답변]: zip파일생성시. [5]파일 다운로드1
2558로손5/4/201513846VB6.0의 ActiveX(OCX) 와 .Net에서 만든 COM DLL의 차이점 ?? [1]
2559로손5/6/201511441    답변글 [답변]: VB6.0의 ActiveX(OCX) 와 .Net에서 만든 COM DLL의 차이점 ??
2557Guest4/24/201510370Serialize , DeSerialize 관련 질문입니다! [3]
1558민식짱4/23/201510639FFmpeg.exe 를 이용한 C# 동영상 인코더 예제를 보고 질문드립니다 [1]파일 다운로드1
1555김응규4/22/201514944WCF net.tcp 관련해서 질문 드립니다. [2]
1556김응규4/22/201512314    답변글 [답변]: WCF net.tcp 관련해서 질문 드립니다. [1]파일 다운로드1
... 46  47  48  49  [50]  51  52  53  54  55  56  57  58  59  60  ...