어떻게:Windows 창 컨트롤 에 스 레 드 보안 호출
26566 단어 창 컨트롤스 레 드 보안 호출
Windows 창 에 접근 하 는 컨트롤 은 본질 적 으로 스 레 드 가 안전 하지 않 습 니 다.두 개 이상 의 스 레 드 가 특정한 컨트롤 을 조작 하 는 상태 가 있 으 면 이 컨트롤 을 일치 하지 않 는 상태 로 만 들 수 있 습 니 다.경쟁 상황 과 잠 금 을 포함 하여 스 레 드 와 관련 된 다른 bug 도 나타 날 수 있 습 니 다.스 레 드 보안 방식 으로 컨트롤 에 접근 하 는 것 이 중요 합 니 다.
.NET Framework 는 비 스 레 드 보안 방식 으로 컨트롤 에 접근 할 때 이 문 제 를 감지 하 는 데 도움 이 된다.디 버 거 에서 프로그램 을 실행 할 때 특정한 컨트롤 의 스 레 드 이외 의 다른 스 레 드 를 만 들 고 이 컨트롤 을 호출 하려 고 하면 디 버 거 는InvalidOperationException메 시 지 를 보 냅 니 다."컨트롤 control name 을 만 드 는 스 레 드 가 아 닙 니 다."
이 이상 은 디 버 깅 기간 과 실행 시의 어떤 상황 에서 신뢰성 있 게 발생 합 니 다.이 오류 정 보 를 표시 할 때 이 문 제 를 복구 하 는 것 을 강력 히 권장 합 니 다..NET Framework 2.0 버 전의.NET Framework 로 작 성 된 프로그램 을 디 버 깅 할 때 이 이상 이 발생 할 수 있 습 니 다.
주의
CheckForIllegalCrossThreadCalls속성의 값 을 false 로 설정 하여 이 이상 을 사용 하지 않 을 수 있 습 니 다.이 컨트롤 은 Visual Studio 2003 에서 와 같은 방식 으로 실 행 됩 니 다.
다음 코드 예제 에 서 는 보조 스 레 드 에서 스 레 드 보안 방식 과 비 스 레 드 보안 방식 으로 Windows 창 컨트롤 을 호출 하 는 방법 을 보 여 줍 니 다.이 는 비 스 레 드 보안 방식 으로TextBox컨트롤 의Text속성 을 설정 하 는 방법 을 보 여 주 며,스 레 드 보안 방식 으로 Text 속성 을 설정 하 는 두 가지 방법 도 보 여 준다.
C#
using System;
using System.ComponentModel;
using System.Threading;
using System.Windows.Forms;
namespace CrossThreadDemo
{
public class Form1 : Form
{
// This delegate enables asynchronous calls for setting
// the text property on a TextBox control.
delegate void SetTextCallback(string text);
// This thread is used to demonstrate both thread-safe and
// unsafe ways to call a Windows Forms control.
private Thread demoThread = null;
// This BackgroundWorker is used to demonstrate the
// preferred way of performing asynchronous operations.
private BackgroundWorker backgroundWorker1;
private TextBox textBox1;
private Button setTextUnsafeBtn;
private Button setTextSafeBtn;
private Button setTextBackgroundWorkerBtn;
private System.ComponentModel.IContainer components = null;
public Form1()
{
InitializeComponent();
}
protected override void Dispose(bool disposing)
{
if (disposing && (components != null))
{
components.Dispose();
}
base.Dispose(disposing);
}
// This event handler creates a thread that calls a
// Windows Forms control in an unsafe way.
private void setTextUnsafeBtn_Click(
object sender,
EventArgs e)
{
this.demoThread =
new Thread(new ThreadStart(this.ThreadProcUnsafe));
this.demoThread.Start();
}
// This method is executed on the worker thread and makes
// an unsafe call on the TextBox control.
private void ThreadProcUnsafe()
{
this.textBox1.Text = "This text was set unsafely.";
}
// This event handler creates a thread that calls a
// Windows Forms control in a thread-safe way.
private void setTextSafeBtn_Click(
object sender,
EventArgs e)
{
this.demoThread =
new Thread(new ThreadStart(this.ThreadProcSafe));
this.demoThread.Start();
}
// This method is executed on the worker thread and makes
// a thread-safe call on the TextBox control.
private void ThreadProcSafe()
{
this.SetText("This text was set safely.");
}
// This method demonstrates a pattern for making thread-safe
// calls on a Windows Forms control.
//
// If the calling thread is different from the thread that
// created the TextBox control, this method creates a
// SetTextCallback and calls itself asynchronously using the
// Invoke method.
//
// If the calling thread is the same as the thread that created
// the TextBox control, the Text property is set directly.
private void SetText(string text)
{
// InvokeRequired required compares the thread ID of the
// calling thread to the thread ID of the creating thread.
// If these threads are different, it returns true.
if (this.textBox1.InvokeRequired)
{
SetTextCallback d = new SetTextCallback(SetText);
this.Invoke(d, new object[] { text });
}
else
{
this.textBox1.Text = text;
}
}
// This event handler starts the form's
// BackgroundWorker by calling RunWorkerAsync.
//
// The Text property of the TextBox control is set
// when the BackgroundWorker raises the RunWorkerCompleted
// event.
private void setTextBackgroundWorkerBtn_Click(
object sender,
EventArgs e)
{
this.backgroundWorker1.RunWorkerAsync();
}
// This event handler sets the Text property of the TextBox
// control. It is called on the thread that created the
// TextBox control, so the call is thread-safe.
//
// BackgroundWorker is the preferred way to perform asynchronous
// operations.
private void backgroundWorker1_RunWorkerCompleted(
object sender,
RunWorkerCompletedEventArgs e)
{
this.textBox1.Text =
"This text was set safely by BackgroundWorker.";
}
#region Windows Form Designer generated code
private void InitializeComponent()
{
this.textBox1 = new System.Windows.Forms.TextBox();
this.setTextUnsafeBtn = new System.Windows.Forms.Button();
this.setTextSafeBtn = new System.Windows.Forms.Button();
this.setTextBackgroundWorkerBtn = new System.Windows.Forms.Button();
this.backgroundWorker1 = new System.ComponentModel.BackgroundWorker();
this.SuspendLayout();
//
// textBox1
//
this.textBox1.Location = new System.Drawing.Point(12, 12);
this.textBox1.Name = "textBox1";
this.textBox1.Size = new System.Drawing.Size(240, 20);
this.textBox1.TabIndex = 0;
//
// setTextUnsafeBtn
//
this.setTextUnsafeBtn.Location = new System.Drawing.Point(15, 55);
this.setTextUnsafeBtn.Name = "setTextUnsafeBtn";
this.setTextUnsafeBtn.TabIndex = 1;
this.setTextUnsafeBtn.Text = "Unsafe Call";
this.setTextUnsafeBtn.Click += new System.EventHandler(this.setTextUnsafeBtn_Click);
//
// setTextSafeBtn
//
this.setTextSafeBtn.Location = new System.Drawing.Point(96, 55);
this.setTextSafeBtn.Name = "setTextSafeBtn";
this.setTextSafeBtn.TabIndex = 2;
this.setTextSafeBtn.Text = "Safe Call";
this.setTextSafeBtn.Click += new System.EventHandler(this.setTextSafeBtn_Click);
//
// setTextBackgroundWorkerBtn
//
this.setTextBackgroundWorkerBtn.Location = new System.Drawing.Point(177, 55);
this.setTextBackgroundWorkerBtn.Name = "setTextBackgroundWorkerBtn";
this.setTextBackgroundWorkerBtn.TabIndex = 3;
this.setTextBackgroundWorkerBtn.Text = "Safe BW Call";
this.setTextBackgroundWorkerBtn.Click += new System.EventHandler(this.setTextBackgroundWorkerBtn_Click);
//
// backgroundWorker1
//
this.backgroundWorker1.RunWorkerCompleted += new System.ComponentModel.RunWorkerCompletedEventHandler(this.backgroundWorker1_RunWorkerCompleted);
//
// Form1
//
this.ClientSize = new System.Drawing.Size(268, 96);
this.Controls.Add(this.setTextBackgroundWorkerBtn);
this.Controls.Add(this.setTextSafeBtn);
this.Controls.Add(this.setTextUnsafeBtn);
this.Controls.Add(this.textBox1);
this.Name = "Form1";
this.Text = "Form1";
this.ResumeLayout(false);
this.PerformLayout();
}
#endregion
[STAThread]
static void Main()
{
Application.EnableVisualStyles();
Application.Run(new Form1());
}
}
}
Windows 창 컨트롤 의 비 스 레 드 보안 호출Windows 창 컨트롤 에 대한 비 스 레 드 보안 호출 방식 은 보조 스 레 드 에서 직접 호출 됩 니 다.프로그램 을 호출 할 때 디 버 거 는 Invalid Operation Exception 을 일 으 켜 컨트롤 에 대한 호출 이 스 레 드 가 안전 하지 않다 고 경고 합 니 다.
C#
// This event handler creates a thread that calls a
// Windows Forms control in an unsafe way.
private void setTextUnsafeBtn_Click(
object sender,
EventArgs e)
{
this.demoThread =
new Thread(new ThreadStart(this.ThreadProcUnsafe));
this.demoThread.Start();
}
// This method is executed on the worker thread and makes
// an unsafe call on the TextBox control.
private void ThreadProcUnsafe()
{
this.textBox1.Text = "This text was set unsafely.";
}
Windows 창 컨트롤 의 스 레 드 보안 호출Windows 창 컨트롤 에 스 레 드 보안 호출
컨트롤 의 속성 을 조회 합 니 다.
Invoke Required 가 true 로 돌아 오 면 실제 호출 컨트롤 의 의뢰 를 사용 하여 호출 합 니 다InvokeRequired.
Invoke Required 가 false 로 돌아 오 면 컨트롤 을 직접 호출 합 니 다.
C#
// This event handler creates a thread that calls a
// Windows Forms control in a thread-safe way.
private void setTextSafeBtn_Click(
object sender,
EventArgs e)
{
this.demoThread =
new Thread(new ThreadStart(this.ThreadProcSafe));
this.demoThread.Start();
}
// This method is executed on the worker thread and makes
// a thread-safe call on the TextBox control.
private void ThreadProcSafe()
{
this.SetText("This text was set safely.");
}
C#
Invoke // This method demonstrates a pattern for making thread-safe// calls on a Windows Forms control. //// If the calling thread is different from the thread that// created the TextBox control, this method creates a// SetTextCallback and calls itself asynchronously using the// Invoke method.//// If the calling thread is the same as the thread that created// the TextBox control, the Text property is set directly. private void SetText(string text){// InvokeRequired required compares the thread ID of the// calling thread to the thread ID of the creating thread.// If these threads are different, it returns true.if (this.textBox1.InvokeRequired){SetTextCallback d = new SetTextCallback(SetText);this.Invoke(d, new object[] { text });}else{this.textBox1.Text = text;}}
BackgroundWorker 를 사용 한 스 레 드 보안 호출
응용 프로그램 에서 다 중 스 레 드 를 실현 하 는 가장 좋 은 방법 은구성 요 소 를 사용 하 는 것 입 니 다.BackgroundWorker 구성 요 소 는 이벤트 구동 모델 을 사용 하여 다 중 스 레 드 를 실현 합 니 다.보조 스 레 드 실행BackgroundWorker이벤트 처리 프로그램,컨트롤 을 만 드 는 스 레 드 실행DoWork과ProgressChanged이벤트 처리 프로그램.DoWork 이벤트 처리 프로그램 에서 컨트롤 을 호출 하지 않도록 주의 하 십시오.
다음 코드 예제 에 서 는 다른 작업 을 수행 하지 않 기 때문에 DoWork 이벤트 처리 프로그램의 실현 이 없습니다.TextBox 컨트롤 의 Text 속성 은 RunWorkerComplete 이벤트 처리 프로그램 에서 직접 설정 합 니 다.
C#
// This event handler starts the form's
// BackgroundWorker by calling RunWorkerAsync.
//
// The Text property of the TextBox control is set
// when the BackgroundWorker raises the RunWorkerCompleted
// event.
private void setTextBackgroundWorkerBtn_Click(
object sender,
EventArgs e)
{
this.backgroundWorker1.RunWorkerAsync();
}
// This event handler sets the Text property of the TextBox
// control. It is called on the thread that created the
// TextBox control, so the call is thread-safe.
//
// BackgroundWorker is the preferred way to perform asynchronous
// operations.
private void backgroundWorker1_RunWorkerCompleted(
object sender,
RunWorkerCompletedEventArgs e)
{
this.textBox1.Text =
"This text was set safely by BackgroundWorker.";
}
Windows 창 에 있 는 ActiveX 컨트롤창 에 ActiveX 컨트롤 을 사용 하면 디 버 거 에서 실 행 될 때 스 레 드 간 Invalid Operation Exception 을 받 을 수 있 습 니 다.이 경우 ActiveX 컨트롤 은 다 중 스 레 드 처 리 를 지원 하지 않 습 니 다.Windows 창 을 사용 하 는 ActiveX 컨트롤 에 대한 더 많은 정 보 는RunWorkerCompleted을 참조 하 십시오.
Visual Studio 를 사용 하면 Visual Studio 숙주 프로 세 스 를 사용 하지 않 아 도 이 이상 을 방지 할 수 있 습 니 다.
신뢰 할 수 있 는 프로 그래 밍
경고
어떤 다 중 스 레 드 를 사용 할 때 코드 는 매우 심각 하고 복잡 한 bug 를 만 들 기 쉽다.더 많은 정보 에 대해 서 는 다 중 스 레 드 를 사용 하 는 모든 해결 방안 을 실현 하기 전에 참조 하 십시오Windows 창 과 비 위탁 관리 프로그램.
참고 하 시 오
퀘 스 트위탁 관리 스 레 드 처리 의 가장 좋 은 방법어떻게:배경 에서 작업 을 실행 합 니까?
참고어떻게:배경 작업 을 위 한 창 구현
기타 자원BackgroundWorker.NET Framework 를 사용 하여 사용자 정의 Windows 창 컨트롤 개발