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)
5621장성욱3/8/20224680c# 로그 관련 질문 [1]
5620김민아3/7/20224702안녕하세요 비관리 객체를 반환하는 메소드 호출 시 궁금한 점이 있어서 질문드립니다 [2]
5619팡팡이3/3/20225978RSA 문의드립니다. [3]
5618김기헌3/2/20224559안녕하세요 생성자 옆에 this 키워드를 붙여 생성자를 여러 개 호출 시 질문드립니다 [2]
5617Edun2/25/20224703ArgumentOutOrRangeException에러 발생 [2]파일 다운로드1
5616csha...2/24/20224596readonly struct로 선언된 구조체를 특정 클래스에서 멤버변수로 가지는 경우 [1]
5615장성욱2/22/20226775SetThreadAffinityMask를 이용한 쓰레드 지정하는 방법에 대해 궁금합니다. [4]
5614민우2/22/20226118SSL 통신 문의 [6]
5613김인태2/22/20224649서버와 PC 간의 어플리케이션 성능 차이 [1]
5612팬입니다2/20/20224462Kastrel 서버 관련 [1]
5611차가워2/19/20224693stopWatch 늘어짐 문의 [3]
5610차가워2/18/20224593Stopwatch 늘어짐 문제 [1]
5609cs린이2/15/20224905c# 함수의 호출 방식에 대해 궁금합니다! [2]
5608지호2/10/20225402시작하세요 C# 8.0 중 제네릭타입의 IEnumerable [3]파일 다운로드1
5607이로운2/10/20224680안녕하세요. 궁금한게 있어서 질문드립니다. [1]
5605강성봉2/10/20224669TCP PSH flag 패킷 수신 에러 [1]
5604LW2/9/20224823VISUAL STUDIO 2019 ==> 2020 설치시 오류가 생겨서 문의드립니다. [3]
5603김진명2/9/20226004C# 10.0 출간은 언제쯤 계획하고 계신가요? [1]
5602신갑영2/8/20224902윈폼에 대해서 질문 드립니다. [1]
5601김인태2/4/20225117setup project 관련 [7]
5600itkim2/3/20225938윈도우 서버 계정 패스워드 인증 문의 [5]
5599레드골드2/3/20226373c#으로 ms word 제어 가능할까요? [6]
5598jaew...2/2/20225189Dictionary는 참조형식인가요?? [1]
5597재원2/2/20225501c# 9.0에 대한 내용을 받을 수 있나요? [1]
5596kss1/26/20225262듀얼 모니터 환경에서 wpf 프로그램 크기 변경 [2]
5594mira...1/25/20224896안녕하세요 try~catch 와 SuspendLayout~ResumeLayout 간 호출 문의드립니다! [2]
1  2  3  4  5  6  7  8  9  10  11  12  [13]  14  15  ...