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)
5594mira...1/25/20224811안녕하세요 try~catch 와 SuspendLayout~ResumeLayout 간 호출 문의드립니다! [2]
5593C#스터디1/12/20225688TaskAwaiter 구현 질문 입니다. [1]
5591유지킴12/24/20215318outofmemory in 32bit [2]파일 다운로드1
5590kss12/23/20214762포인터 메모리 에러 [1]
5589초급12/22/20214979c# -> 라즈베리파이(db 접속)시 에러 발생 [7]
5588김지신12/21/20214901안녕하세요 String 변수 참조 주소 질문입니다. [2]
5587이완호12/17/20216007C# SharpDX 화면 캡쳐 관련해서 질문 드립니다. [1]
5586조미김12/16/20215201윈폼에서 메인 스레드와 UI 스레드의 차이점 질문드립니다 [4]
5585김준희12/16/20216008C# 윈폼 TCP/IP 데이터 연속으로 보낼때 [1]
5583난인간이다12/10/20214985.net 6 dynamic pgo 활성화 및 예상되는 문제 [1]
5582김준희12/9/202112408C# TCP/IP 통신시 연결 끊김 에러 [1]
5581김시준12/9/20215534닷넷에서 파일 delete 함수는 왜 비동기가 없는 것인가요? [2]
5580카짜프로...12/7/20216713패턴매칭 -튜퓰비교에 관한 오류사항과 궁금증 [1]파일 다운로드1
5579카짜프로...12/6/2021607911.12 메서드 중복정의에 대한 질문 [1]
5577감사합니...11/30/20215085visual studio 2015 update 3를 다운받을려고 하는데 x64/x86 차이점이 뭘까요? [2]
5576노홍구11/29/20214972C# 으로 USB 스캐너 프린터 리셋하기 입니다. [1]
5575베라11/23/20215020event handler 관련 문의 [2]
5574박원웅11/22/20215352닷넷 프레임워크 산출물의 배포시 해당 환경에 프레임워크 버전이 설치되어 있지 않는 경우를 고려한 배포방법은? [3]
5573mijin11/21/20215169System.NullReferenceException 에 대한 질문 [1]
5572김현진11/21/20215338C# list.Clear() 호출에 대한 문의 드립니다. [4]
5571한예지 donator11/18/20215274무설치 프로그램 원리가 궁금합니다. [2]
5570초보11/16/20215802주식데이터 초당 수신 건수를 구하고 있는데 처리속도가 느려서요 [2]
5569카짜프로...11/14/20216609381페이지 UTC에대한 언급이 그리니치 천문대 시간으로 되어있는게 맞나요? [1]
5568카짜프로...11/14/20216668그림 5.20, 그림 5.22 언급 오류 [1]
5567Edun11/2/20215614쿼리문을 코드로 어떻게 처리할 수 있을까요? [2]
5566민성10/26/20215431Linq에 관해서 [1]
1  2  3  4  5  6  7  8  9  10  11  12  13  [14]  15  ...