Microsoft MVP성태의 닷넷 이야기
글쓴 사람
한예지 donator
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

선생님 안녕하세요.
Thread.Sleep(500), await Task.Delay(500), Task.Delay(500) 차이점이 궁금해서
선생님께서 작성하신 글들을 바탕으로 이해한 것을 정리했는데 잘못된 점이 있는지 확인 부탁드립니다.
항상 감사드립니다!

[#00] 예제
using System;
using System.Diagnostics;
using System.Threading;
using System.Threading.Tasks;
using System.Windows.Forms;

// 윈폼에 TextBox(MultLine), Button을 하나 추가
namespace WindowsFormsApplication12
{
    public partial class Form1 : Form
    {
        public Form1()
        {
            InitializeComponent();
        }

        private void button1_Click(object sender, EventArgs e)
        {
            Stopwatch sw = new Stopwatch();
            sw.Start();
            var x = countDonw();
            var y = countDonw();
            Task.WhenAll(x, y);
            sw.Stop();
            MessageBox.Show(Math.Round(sw.Elapsed.TotalSeconds, 3) + "초");
        }

        private async Task countDonw()
        {
            for (int i = 9; i >= 0; i--)
            {
                textBox1.Text += i.ToString();
               
        // case ① ★
                Thread.Sleep(500);

        // case ② ★
        // await Task.Delay(500);

        // case ③ ★
        // Task.Delay(500);
            }
        }
    }
}
[출력]
case ①인 경우 : 버튼을 누르는 순간 UI freeze가 발생하고 대략 10초 후에 텍스트 박스에 "98765432109876543210"가 표시되고 MessageBox도 띄워집니다.
case ②인 경우 : 버튼을 누르자마자 UI freeze 없이 "바로" 텍스트 박스에는 98765432109876543210"가 표시되고 MessageBox 박스도 띄워집니다.
case ③인 경우 : 버튼을 누르자마자 UI freeze 없이 "바로" 텍스트 박스에는 "99"가 표시되고 MessageBox 박스도 띄어집니다.
             그리고 0.5초 뒤에 텍스트 박스에 "88"가 추가된다. 이런 식으로 0.5초마다 텍스트 박스에 숫자가 표시되면
             텍스트 박스에는 0.5초씩 5초 동안 "99887766554433221100"가 최종적으로 표시됩니다.


[#01] case ① 설명(★ 표시된 부분)
async 키워드를 사용했지만 별도의 스레드를 생성하는 코드가 없기 때문에 동기적으로 코드가 실행됩니다.
var x = countDonw(); 실행 후 var y = countDonw();가 실행됩니다.
따라서 98...0 → 987...0 순서로 출력이 됩니다.
참고로 Thread.Sleep을 이용해 메시지 루프 실행을 막고 있기 때문에
버튼을 눌렸을 때 TextBox에는 여전히 빈 문자열만 출력이 되고
화면은 대략 10초 정도 멈춘 후에야 "98765432109876543210"가 출력됩니다.

여기서 궁금한 점은
https://www.sysnet.pe.kr/2/0/12544?pageno=5 글에는
"Invalidate를 호출해 WM_PAINT를 전송했으므로 이것이 처리되기 위해서는 반드시 메시지 루프가 돌아야 한다." 라는 문장이 있습니다.
WM_PAINT은 textBox1에 "98765432109876543210" 그리라는 메시지 같은데
this.textBox1.Invalidate();를 제가 호출하지 않았음에도 어떻게 WM_PAINT를 메시지 큐에 넣었는지 궁금합니다.
Form1.Designer.cs 파일에서 Invalidate를 검색했는데 보이지가 않아서 혹시 컴파일러가 자동으로 만들어줄까요?


[#02] case ② 설명(★ 표시된 부분)
[코드 블록 ㉠]
await Task.Delay(5초);
[코드 블록 ㉡]

보통은 5초 후에 스레드 풀에 스레드를 가져와서 [코드 블록 ㉡]을 실행하지만
[#00]처럼 SynchronizationContext가 제공되는 WinForm이라면
await 이후 작업([코드 블록 ㉡])은 스레드 풀에 스레드가 아닌 UI 스레드가 담당하기 때문에
invoke 메서드를 사용할 필요가 없습니다.

Task.Delay(5초)는 내부적으로 System.Timers.Timer를 사용하는데 기능적으로 아래와 동일하다고 볼 수 있다.
(https://www.sysnet.pe.kr/2/0/13069 참고했습니다.)
timer.Elapsed += delegate { [코드 블록 ㉡] };
timer.Start();

이때 스레드 풀에 스레드가 아닌 UI 스레드가 담당하기 때문에 바로 위에서 보여준 코드를 아래와 같이 수정할 수 있다.
(https://www.sysnet.pe.kr/2/0/13191 참고했습니다.)

SynchronizationContext uiContext = SynchronizationContext.Current;
timer.Elapsed += delegate
{
    // 코드 블록 ㉡을 UI 스레드에서 실행
    uiContext.Post(() =>
    {
        // 코드 블록 ㉡을 여기에 작성
    }, null);
};
timer.Start();


[#03] case ③ 설명(★ 표시된 부분)
[코드 블록 ㉠]
Task.Delay(5초);
[코드 블록 ㉡]

만약 await 키워드가 없다면
타이머의 Elapsed 이벤트에 [코드 블록 ㉡]을 이벤트 핸들러로 등록시키지 않습니다.
이때는 스레드 풀에 스레드가 아닌 UI 스레드로 [코드 블록 ㉡]을 "바로" 실행시킵니다.
사실, await 키워드가 없기 때문에 사실 Task.Delay(5초)는 없어도 되는 코드입니다.


정리하면
[질문 ①]은
this.textBox1.Invalidate();를 제가 호출하지 않았음에도 어떻게 WM_PAINT를 메시지 큐에 넣었는지 궁금합니다.
Form1.Designer.cs 파일에서 Invalidate를 검색했는데 보이지가 않아서 혹시 컴파일러가 자동으로 만들어줄까요?

[질문 ②]는
[#00] 예제를 바탕으로 [#01](case ①), [#02](case ②), [#03](case ③)을 정리했는데
혹시 제가 잘못 이해한 부분이 있을까요?


[연관 글]






[최초 등록일: ]
[최종 수정일: 9/3/2023]


비밀번호

댓글 작성자
 



2023-09-03 09시55분
[답변1] 해당 코드에서 Invalidate는 물론 사용자가 직접 처리해도 되겠지만 "textBox1.Text += i.ToString();" 코드의 Text set 메서드에서 내부적으로 처리하고 있기 때문에 그렇게 동작하는 것입니다.

[답변2] 전반적으로 이해하신 것이 맞습니다. 하지만, 코드 자체가 매끄럽지 않는 부분이 있는데 관련해서는 아래의 글에 별도로 정리를 했으니 참고하세요.

C# - async 메서드 호출 원칙
; https://www.sysnet.pe.kr/2/0/13405
정성태
2023-09-04 01시29분
쉽게 설명해 주셔서 감사합니다^^

선생님 한 주 파이팅 하시라고 커피 보냈습니다!
한예지

1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...
NoWriterDateCnt.TitleFile(s)
5566민성10/26/20215504Linq에 관해서 [1]
5565pass...10/25/20216777wpf에서 Cefsharp를 AnyCPU로 작업 했는데 실행이 되지 않습니다. [4]
5564초보10/21/20215077오버라이드 관련 질문드립니다 [2]
5562서지훈10/20/20215057win32 api 문의 드립니다. (EnableMenuItem) [5]
5561Edun10/15/20215531도와주세요!! 팝업 띄우는 더블클릭 이벤트!! 부탁드립니다 ㅠㅠ [4]파일 다운로드1
5560Edun10/13/20215692C#초보자 개념이해 도와주세요!! [4]
5559최가운10/5/20215999C# debugging 중에 Thread Pool Worker들이 사라지는데요 [3]
5558옥코드9/29/20216229안녕하세요! EntityFramework Linq 질문 드리겠습니다. [4]
5557임기성9/16/20216783웹 사이트(IIS)가 오류로 인하여 빈번하게 죽는 이슈 문의 [1]
5556hjlee9/13/20215585visual studio 설치 없이 csi.exe (C# interactive) tool을 설치할 수 있을까요? [2]
5555이성열 donator9/2/20215384C# 런타임(Reflection)에서 소스코드에 입력한 함수설명(Summary)을 가져올 수 있을까요? [3]
5554농상9/2/20216106enum 원소값이 중복인 경우 출력 [4]
5553shdt...8/31/20215050제너릭을 new로 할당했는데 null인 경우가 존재하나요? [1]
5552hjsh...8/30/20215314C# 폼 - 마우스가 다른 버튼 위를 움직일 때도 폼의 페인트핸들러가 호출되는 것이 맞나요? [2]
5551하나다라마8/27/20215446뽀모도로 타이머 개선사항 [1]
5550하영8/27/20216518SqlDataReader 관련 질문입니다. [8]
5549책 감사...8/23/20217909C# 9.0 책 오타 [1]
5548Dev8/23/20216295String의 IndexOfAny 함수 질문 [3]
5547민성8/20/20215663WPF Radio 버튼을 GroupName으로 읽어와서 비교하기 [1]
5546Deve...8/19/20216521C++ C# API 연동 관련 [1]
5545영귤8/18/20215486C# 14.3장 예제에서 반환 값의 값 복사 부하가 어디서 발생하는 것인가요? [4]
5544이상호8/18/20215803C# Graphics 객체 저장 가능 여부 [2]
5543정형지8/18/20215566c# outofmemoryexception [3]
5542영귤8/11/20218200C#9.0 886쪽 오류 발생 안 해요 [2]
5541서지훈8/11/20219080C# 응용프로그램(Winform)의 비정상 종료(Exit Code) [2]
5540산들마을...8/11/20216527C# 많은 Control(Label 300개) Update 관련 문의 [2]파일 다운로드1
1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...