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분
쉽게 설명해 주셔서 감사합니다^^

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

... 16  17  18  19  20  21  22  23  24  25  26  [27]  28  29  30  ...
NoWriterDateCnt.TitleFile(s)
5219티지레몬9/9/201910045c# PCB 자동화 프로그램(윈도우 폼 위주로 작업) 제작 준비 [3]
5218민성9/9/20198101안녕하세요 WPF에서 xaml 안에 다른 xaml을 넣고 싶습니다. [1]파일 다운로드1
5216WPF9/8/20199199WPF에서 XAML Islands를 사용하여 Win2D를 사용하니 그래픽 품질이 저하됩니다. [2]파일 다운로드1
5215허송세월9/5/20199002중복실행 방지 관련 문의 [2]파일 다운로드1
5214Jang...9/4/20198637[DB 테이블의 데이터 변경에 대한 알림 처리] SQL-Server말고 MySQL은 불가능하겠죠? [1]
5213진우8/31/20197871c# 람다 변수 캡쳐 문의 [2]
5212심성보8/29/20199582Clipboard내 여러개의 이미지를 PictureBox로 불러오는 문제 [2]
5211최휘철8/24/20198672CLR20r3 관련된 윈도우 오류입니다. ㅠㅠ 도와주세요. / 아래글 관련하여 관련 파일 올려 드려요^^ [1]파일 다운로드1
5210최휘철8/23/201912660CLR20r3 관련된 윈도우 오류입니다. ㅠㅠ 도와주세요. [5]
5209세퉁8/21/20198272폰트 파일 속성 값을 가져오는 방법 질문 드립니다. [2]파일 다운로드1
5208홍길동8/19/20198954DebugDiag에서 .Net의 Stack Trace를 Windbg에서는 어떻게 볼 수 있나요? [3]
5207민성8/16/20197342네 소스 전체를 올리도록 하겠습니다. [2]파일 다운로드1
5206민성8/14/20197475전 재현 가능하다고 봤는데 다시올리도록 하겠습니다. [1]
5205miny...8/14/20198248안녕하세요 .WPF ListBox시 체크박스가 있는데 체크박스에서 체크가 되었는지 알수 있는 방법이 있을까요? [1]
5204영민8/8/201911027안녕하세요 디버깅시 콘솔창을 띠어서 볼수가 없나요? [7]
5202민성8/6/20197786WPF에서 <Application.Resources에 xaml에 있는 icon 값을 저장하고 xaml에 불러다 사용하고 싶은데요 [1]
5201김대훈8/3/20197514상속시 생성자에 대해 질문드립니다 [3]
5200농상7/30/201910251foreach로 데이터 변경 [2]
5190오리다람7/20/20197626질문드립니다. [3]
5189진우7/19/20198205C# 스레드풀 코어별 실행 문의 [2]
5188황태관7/19/20197377비주얼베이직 2019 실행 할때 마다.. [3]
5187플하7/19/201910189UWP 관련 궁금한 사항에 대해서 [1]
5186김대훈7/14/20198670박싱과 언박싱에 대해 [2]
5185농상7/13/20197598Nullable에 대해서 [1]
5184김대훈7/4/20197416저자님의 책을 다 본후에는 [2]
51837/2/20198166.NET Compact Freamwork 컨트롤러 더블버퍼링 [1]
... 16  17  18  19  20  21  22  23  24  25  26  [27]  28  29  30  ...