성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] VT sequences to "CONOUT$" vs. STD_O...
[정성태] NetCoreDbg is a managed code debugg...
[정성태] Evaluating tail call elimination in...
[정성태] What’s new in System.Text.Json in ....
[정성태] What's new in .NET 9: Cryptography ...
[정성태] 아... 제시해 주신 "https://akrzemi1.wordp...
[정성태] 다시 질문을 정리할 필요가 있을 것 같습니다. 제가 본문에...
[이승준] 완전히 잘못 짚었습니다. 댓글 지우고 싶네요. 검색을 해보...
[정성태] 우선 답글 감사합니다. ^^ 그런데, 사실 저 예제는 (g...
[이승준] 수정이 안되어서... byteArray는 BYTE* 타입입니다...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <h1 style='font-family: Malgun Gothic, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>두 개의 윈도우를 각각 실행하는 방법(Windows Forms, WPF)</h1> <p> 아래와 같은 질문이 있군요.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > WPF에서 로딩중 이미지를 구현 ; <a target='tab' href='http://www.sysnet.pe.kr/3/0/5104'>http://www.sysnet.pe.kr/3/0/5104</a> ; <a target='tab' href='http://www.sysnet.pe.kr/3/0/5107'>http://www.sysnet.pe.kr/3/0/5107</a> </pre> <br /> 해당 문제를 간략하게 정리해 보면, 다음과 같이 Main Form에서 스레드가 어떤 작업을 하는 사이,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System.Windows.Forms; namespace WindowsFormsApp1 { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void button1_Click(object sender, System.EventArgs e) { WaitForm waitForm = new WaitForm(); waitForm.Show(); // ...[생략: 사용자 작업]... } } } </pre> <br /> WaitForm에서는 사용자로 하여금 대기하라는 표현을 하고 싶은 것입니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > using System; using System.Windows.Forms; namespace WindowsFormsApp1 { public partial class WaitForm : Form { public WaitForm() { InitializeComponent(); } Timer _timer; int _wait = 10; private void WaitForm_Load(object sender, EventArgs e) { _timer = new Timer(); _timer.Interval = 1000; _timer.Tick += _timer_Tick; _timer.Start(); } private void _timer_Tick(object sender, EventArgs e) { this.Text = _wait.ToString(); _wait--; if (_wait < 0) { _timer.Stop(); this.Close(); } } } } </pre> <br /> 기대하기로는, MainForm에서 작업을 하는 사이 별도로 뜬 WaitForm은 10, 9, 8, ...로 화면에 카운팅이 되도록 만들려는 것입니다. 간단한 재현을 위해 MainForm의 버튼 이벤트에 다음과 같이 Thread.Sleep을 추가할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > private void button1_Click(object sender, System.EventArgs e) { WaitForm waitForm = new WaitForm(); waitForm.Show(); Thread.Sleep(1000 * 10); } </pre> <br /> 그런데, 실제로 위의 코드를 실행해 보면 MainForm에서 10초 동안 멈춰있는 사이 WaitForm의 동작도 멈추게 됩니다. 왜 그럴까요?<br /> <br /> <hr style='width: 50%' /><br /> <br /> 이유는 스레드가 하나이기 때문입니다. WaitForm을 Show한 Thread는 이후 Thread.Sleep의 실행으로 10초 동안 CPU로부터 아무런 자원 할당을 받지 못합니다. (또는, 작업을 한다고 해도 CPU는 그 작업만을 할 뿐입니다.) 이로 인해, WaitForm 내부의 코드는 전혀 실행하지 못하게 되고 자연스럽게 화면은 멈춰버립니다.<br /> <br /> 이 현상을 해결하려면 WaitForm 윈도우를 별도의 스레드에서 동작시켜야 합니다. 이에 대해서는 전에 다음의 글로 한번 다룬 적이 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > C# - 작업자 스레드와 UI 스레드 ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/11287'>http://www.sysnet.pe.kr/2/0/11287</a> </pre> <br /> 따라서 다음과 같이 코드 변경을 할 수 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void button1_Click(object sender, System.EventArgs e) { <span style='color: blue; font-weight: bold'>Thread waitThread = new Thread(threadFunc); waitThread.Start();</span> Thread.Sleep(1000 * 10); } <span style='color: blue; font-weight: bold'>private void threadFunc() { WaitForm waitForm = new WaitForm(); Application.Run(waitForm); }</span> } </pre> <br /> 이제 다시 실행을 하면, MainForm의 경우 Thread.Sleep으로 멈춰 있는 사이 WaitForm은 별도의 스레드를 할당받았기 때문에 정상적으로 카운팅 코드가 실행됩니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 그런데, 여기서 주의할 점이 있습니다. MainForm과 WaitForm을 생성하는 스레드가 달라졌으므로 서로의 UI 객체를 직접 건드려서는 안 된다는 것인데, 이에 대해서는 예전에도 정리한 적이 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > UI 요소의 접근은 반드시 그 UI를 만든 스레드에서! ; <a target='tab' href='http://www.sysnet.pe.kr/2/0/11561'>http://www.sysnet.pe.kr/2/0/11561</a> </pre> <br /> 따라서 threadFunc에 다음과 같은 식의 코드를 추가하면,<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > private void threadFunc() { <span style='color: blue; font-weight: bold'>this.Text = "TEST";</span> WaitForm waitForm = new WaitForm(); Application.Run(waitForm); } </pre> <br /> 이런 예외가 발생합니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > System.InvalidOperationException HResult=0x80131509 Message=Cross-thread operation not valid: Control 'Form1' accessed from a thread other than the thread it was created on. Source=System.Windows.Forms StackTrace: at System.Windows.Forms.Control.get_Handle() at System.Windows.Forms.Control.set_WindowText(String value) at System.Windows.Forms.Form.set_WindowText(String value) at System.Windows.Forms.Control.set_Text(String value) at System.Windows.Forms.Form.set_Text(String value) at WindowsFormsApp1.Form1.threadFunc() in F:\...\WindowsFormsApp1\Form1.cs:line 24 at System.Threading.ThreadHelper.ThreadStart_Context(Object state) at System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx) at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx) at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state) at System.Threading.ThreadHelper.ThreadStart() </pre> <br /> 해결 방법은 "<a target='tab' href='http://www.sysnet.pe.kr/2/0/11561'>UI 요소의 접근은 반드시 그 UI를 만든 스레드에서!</a>" 글에서 설명했으니 생략합니다.<br /> <br /> <hr style='width: 50%' /><br /> <br /> 그렇다면 동일한 코드를 WPF에서는 어떻게 해야 할까요? 단지 새롭게 도입한 Dispatcher 모델을 사용해야 한다는 정도만 차이가 있습니다.<br /> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // MainWindow.xaml.cs using System.Threading; using System.Windows; namespace WpfApp1 { public partial class MainWindow : Window { public MainWindow() { InitializeComponent(); } private void Button_Click(object sender, RoutedEventArgs e) { Thread waitThread = new Thread(threadFunc); waitThread.SetApartmentState(ApartmentState.STA); waitThread.Start(); Thread.Sleep(1000 * 10); } private void threadFunc() { WaitForm waitForm = new WaitForm(); waitForm.Show(); <span style='color: blue; font-weight: bold'>System.Windows.Threading.Dispatcher.Run();</span> } } } </pre> <br /> <pre style='margin: 10px 0px 10px 10px; padding: 10px 0px 10px 10px; background-color: #fbedbb; overflow: auto; font-family: Consolas, Verdana;' > // WaitForm.xaml.cs using System; using System.Windows; namespace WpfApp1 { public partial class WaitForm : Window { public WaitForm() { InitializeComponent(); <span style='color: blue; font-weight: bold'>_timer = new System.Windows.Threading.DispatcherTimer();</span> _timer.Tick += _timer_Tick; _timer.Interval = new TimeSpan(0, 0, 1); _timer.Start(); } private void _timer_Tick(object sender, EventArgs e) { this.Title = _wait.ToString(); _wait--; if (_wait < 0) { _timer.Stop(); this.Close(); } } protected override void OnClosed(EventArgs e) { base.OnClosed(e); <span style='color: blue; font-weight: bold'>System.Windows.Threading.Dispatcher.CurrentDispatcher.InvokeShutdown();</span> } System.Windows.Threading.DispatcherTimer _timer; int _wait = 10; } } </pre> <br /> (<a target='tab' href='https://www.sysnet.pe.kr/bbs/DownloadAttachment.aspx?fid=1414&boardid=331301885'>첨부 파일은 이 글의 예제 프로젝트를 포함</a>합니다.)<br /> </p><br /> <br /><hr /><span style='color: Maroon'>[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]</span> </div>
첨부파일
스팸 방지용 인증 번호
2146
(왼쪽의 숫자를 입력해야 합니다.)