Microsoft MVP성태의 닷넷 이야기
교재 689, 690쪽(async/await) 질문입니다. [링크 복사], [링크+제목 복사],
조회: 12178
글쓴 사람
한예지 donator
홈페이지
첨부 파일
 

선생님 안녕하세요.

https://www.sysnet.pe.kr/3/0/5314 에 글을 읽고 좀 더 명확하게 정리했봤는데
혹시 제가 잘못 해석한 부분이 있을까요?

private static async void ProcessTcpClient(TcpClient client){
    NetworkStream ns = client.GetStream();
    
    byte[] buffer = new byte[1024];
    // await 키워드를 만나면 Main 스레드는 ProcessTcpClient 메서드를 벗어난다.
    // 스레드 풀에서 가져온 A 스레드를 이용해서 작업
    // IRP(I/O 요청 패킷)을 디바이스 드라이버의 IRP 큐에 IRP를 큐잉작업까지만 스레드 A는 수행하고
    // 스레드 A는 스레드 풀에서 회수한다.
    int received = await ns.ReadAsync(buffer, 0, buffer.Length);
    // 만약 Read 작업 결과이 완료되었다면 스레드 풀에서 스레드를 꺼내서 아래 두 줄을 실행하는데
    // 이때 우연히 스레드 A를 꺼낼 수 있거나 혹은 새로운 스레드 B를 꺼낼 수 있다.
    // 하지만 여기서는 스레드 B를 통해서 아래 2줄을 수행한다고 가정한다.
    string txt = Encoding.UTF8.GetString(buffer, 0, buffer.Length);
    byte[] sendBuffer = Encosing.UTF8.GetBytes("Hello :" + txt);
    
    // await 키워드를 만났기 때문에 이전에 수행된 스레드 B가 스레드 풀에서 회수되고
    // 스레드 풀에서 꺼낸 새로운 스레드 C를 통해 작업을 수행한다.
    // 마찬가지로 디바이스 드라이버의 IRP 큐에 IRP를 큐잉작업까지만 스레드 C는 수행하고
    // 스레드 풀에서 스레드 C를 회수한다.
    await ns.WriteAsync(sendBuffer, 0, sendBuffer.Length);
    // 스레드 Write 작업이 완료되었다면 스레드 풀에서 스레드를 꺼내서 아래 한 줄을 수행하는데
    // 이때 우연히 스레드 A 혹은 스레드 B를 꺼낼 수 있지만
    // 여기서는 스레드 D를 통해서 아래 한 줄을 수행한다고 가정한다.
    ns.Close();
}








[최초 등록일: ]
[최종 수정일: 5/9/2022]


비밀번호

댓글 작성자
 



2022-05-09 10시32분
아래와 같이 정정했으니 또 궁금하신 것이 있으면 질문해주세요.

private static async void ProcessTcpClient(TcpClient client){
    NetworkStream ns = client.GetStream();
    
    byte[] buffer = new byte[1024];
    // await 키워드를 만나면 Main 스레드는 ProcessTcpClient 메서드를 벗어난다.
       ==> 엄밀히는 그럴 수도 있고, 아닐 수도 있습니다. (await 대상이 되는 메서드가 꼭 내부적으로 비동기 작업을 한다는 보장은 없습니다. (참고: https://www.sysnet.pe.kr/2/0/11431)
       
    // 스레드 풀에서 가져온 A 스레드를 이용해서 작업
       ==> 아직 이 단계에서는 스레드 풀에서 가져온 스레드로 작업을 하지 않습니다.
       ==> ReadAsync 내부의 비동기 호출 단계까지는 "ProcessTcpClient" 메서드를 호출한 스레드가 담당합니다.

    // IRP(I/O 요청 패킷)을 디바이스 드라이버의 IRP 큐에 IRP를 큐잉 작업까지만 호출 스레드가 담당
    int received = await ns.ReadAsync(buffer, 0, buffer.Length);

    // 만약 Read 작업 결과가 완료되었다면 스레드 풀에서 스레드(예를 들어 스레드 A)를 꺼내서 아래 두 줄을 실행하는데,
    string txt = Encoding.UTF8.GetString(buffer, 0, buffer.Length);
    byte[] sendBuffer = Encosing.UTF8.GetBytes("Hello :" + txt);
    
    // 마찬가지로 디바이스 드라이버의 IRP 큐에 IRP를 큐잉 작업까지만 스레드 A는 수행하고
    await ns.WriteAsync(sendBuffer, 0, sendBuffer.Length);
    // 스레드 Write 작업이 완료되었다면 스레드 풀에서 스레드를 꺼내서 아래 한 줄을 수행하는데
    // 이때 우연히 스레드 A를 꺼낼 수 있지만
    // 여기서는 또다른 스레드 B를 통해서 아래 한 줄을 수행한다고 가정한다.
    ns.Close();
}
정성태
2022-05-10 01시18분
[한예지] 답변 감사드립니다!

답변해 주신 내용을 바탕으로 아래와 같이 정리해도 문제 없을까요?

private static async void ProcessTcpClient(TcpClient client){
    NetworkStream ns = client.GetStream();
    
    byte[] buffer = new byte[1024];
    // Main 스레드가 IRP(I/O 요청 패킷)을 디바이스 드라이버의 IRP 큐에 IRP를 큐잉 작업까지
    // 처리하고 ProcessTcpClient 메서드를 벗어난다.
    int received = await ns.ReadAsync(buffer, 0, buffer.Length);

    // 만약 Read 작업 결과가 완료되었다면 스레드 풀에서 스레드(스레드 A라고 가정)를 꺼내서 아래 두 줄을 실행하는데,
    string txt = Encoding.UTF8.GetString(buffer, 0, buffer.Length);
    byte[] sendBuffer = Encosing.UTF8.GetBytes("Hello :" + txt);
    
    // 마찬가지로 디바이스 드라이버의 IRP 큐에 IRP를 큐잉 작업까지만 스레드 A는 수행하고 스레드 풀에 회수된다.
    await ns.WriteAsync(sendBuffer, 0, sendBuffer.Length);
    // 스레드 Write 작업이 완료되었다면 스레드 풀에서 스레드를 꺼내서 아래 한 줄을 수행하는데
    // 이때 우연히 스레드 A를 꺼낼 수 있지만
    // 여기서는 또 다른 스레드 B를 통해서 아래 한 줄을 수행한다고 가정한다.
    ns.Close();
}
[guest]
2022-05-10 01시34분
넵, 개념적으로 그렇게 이해하시면 좋겠습니다. (그나저나... 많은 분들이 이에 대해 혼란을 느끼시는 듯하니 관련해서 글을 하나 써야겠군요. ^^;)
정성태
2022-05-10 04시03분
[한예지] 작성해 주시면 정말 감사하죠! 기대하고 있겠습니다^^

아, 혹시 후원 채널을 1개로 하신 이유가 있으신가요??

예전에 있던 BUY ME A COFFEE 버튼은 회원가입을 안 해도 되는데

paypal은 회원가입을 해야 되네요 ㅜ_ㅜ
[guest]
2022-05-10 07시06분
아쉽게도 donaricano("https://donaricano.com/") 서비스가 중단되었습니다. ^^ (게다가 국내에서는 마땅한 기부 링크가 없습니다.)
정성태

... 91  92  93  [94]  95  96  97 
NoWriterDateCnt.TitleFile(s)
111정성태3/19/20059775    답변글 [답변]: 닷넷 프레임워크 배포
107안연준3/17/20059021[-_-]스마트 클라이언트에 관련 된 질문[-_-]
110정성태3/19/20059829    답변글 [답변]: [-_-]스마트 클라이언트에 관련 된 질문[-_-]
116안연준3/21/20059409        답변글 [답변]: 친절한 답변 고맙습니다.
98김용국3/18/200511174정성태님... 연결이 준비됬습니다
99정성태3/2/200512067    답변글 [답변]: 정성태님... 연결이 준비됬습니다
100김용국3/18/20058906        답변글 [답변]: 죄송합니다! 급히올리느라 file path의 변경을 하지않고 그냥올렸네요....!
101정성태3/2/20059158            답변글 [답변]: [답변]: 죄송합니다! 급히올리느라 file path의 변경을 하지않고 그냥올렸네요....!
102김용국3/18/20059213                답변글 [답변]: 거듭죄송하네요..... 후~~~주소를 변경하고 빌드를 다시 했습니다....
103정성태3/2/20059109                    답변글 [답변]: [답변]: 거듭죄송하네요..... 후~~~주소를 변경하고 빌드를 다시 했습니다....
104김용국3/2/20059320                        답변글 [답변]: 네... 빨리 검토해 봐 주셔서 감사합니다~~
105정성태3/2/20058550                            답변글 [답변]: [답변]: 네... 빨리 검토해 봐 주셔서 감사합니다~~
106김용국3/2/20059021                                답변글 [답변]: 답변감사합니다!
91김용국2/28/200510833IE에 WindowsFormControl을 올려 실행하면 이런에러가 나네요???파일 다운로드1
92정성태2/28/200512138    답변글 [답변]: IE에 WindowsFormControl을 올려 실행하면 이런에러가 나네요???
93김용국2/28/200511078        답변글 [답변]: 답변감사합니다... 재질문을 드립니다
94정성태2/28/200511659            답변글 [답변]: [답변]: 답변감사합니다... 재질문을 드립니다
95김용국2/28/200511734                답변글 [답변]: IE주소창에서 해당주소로 실행을 해보니....디버깅 PopUp화면이...^
96정성태2/28/200510789                    답변글 [답변]: [답변]: IE주소창에서 해당주소로 실행을 해보니....디버깅 PopUp화면이...^
97김용국3/1/200511318                        답변글 [답변]: 준비되는데로 말씀드리겠습니다 ^^
88안지환2/22/200512863^^ 사이트 잘 들러보았습니다.
89정성태2/22/200512833    답변글 [답변]: ^^ 사이트 잘 들러보았습니다.
85한기열2/22/200511445정성태님 홈같은 부드러운 페이지 넘김?은 어떻게 구현하나요?
86정성태2/22/200512159    답변글 [답변]: 정성태님 홈같은 부드러운 페이지 넘김?은 어떻게 구현하나요? [2]
84김용국2/21/200512227Smart Client에 관한 문의 드립니다.
87정성태2/22/200512289    답변글 [답변]: Smart Client에 관한 문의 드립니다.
... 91  92  93  [94]  95  96  97