[C#/WINFORM/.NET6] ISynchronizeInvoke 인터페이스 : InvokeRequired 코드 패턴 자동화하기
■ ISynchronizeInvoke 인터페이스를 사용해 InvokeRequired 코드 패턴을 자동화하는 방법을 보여준다. ▶ SynchronizeInvokeExtension.cs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 |
using System.ComponentModel; namespace TestProject; /// <summary> /// 호출 동기화 확장 /// </summary> public static class SynchronizeInvokeExtension { //////////////////////////////////////////////////////////////////////////////////////////////////// Method ////////////////////////////////////////////////////////////////////////////////////////// Static //////////////////////////////////////////////////////////////////////////////// Public #region 필요시 호출하기 - InvokeIfRequired<TTarget>(targetObject, action) /// <summary> /// 필요시 호출하기 /// </summary> /// <typeparam name="TTarget">타겟 객체 타입</typeparam> /// <param name="targetObject">타겟 객체</param> /// <param name="action">액션</param> public static void InvokeIfRequired<TTarget>(this TTarget targetObject, Action<TTarget> action) where TTarget : ISynchronizeInvoke { if(targetObject.InvokeRequired) { targetObject.Invoke(action, new object[] { targetObject }); } else { action(targetObject); } } #endregion #region 필요시 호출하기 - InvokeIfRequired<TTarget, TResult>(targetObject, function) /// <summary> /// 필요시 호출하기 /// </summary> /// <typeparam name="TTarget">타겟 객체 타입</typeparam> /// <typeparam name="TResult">결과 타입</typeparam> /// <param name="targetObject">타겟 객체</param> /// <param name="function">함수</param> /// <returns>처리 결과</returns> public static TResult InvokeIfRequired<TTarget, TResult>(this TTarget targetObject, Func<TTarget, TResult> function) where TTarget : ISynchronizeInvoke { return targetObject.InvokeRequired ? (TResult)targetObject.Invoke(function, new object[] { targetObject }) : function(targetObject); } #endregion } |
▶ MainForm.cs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 |
namespace TestProject { /// <summary> /// 메인 폼 /// </summary> public partial class MainForm : Form { //////////////////////////////////////////////////////////////////////////////////////////////////// Constructor ////////////////////////////////////////////////////////////////////////////////////////// Public #region 생성자 - MainForm() /// <summary> /// 생성자 /// </summary> public MainForm() { InitializeComponent(); this.runButton.Click += runButton_Click; } #endregion //////////////////////////////////////////////////////////////////////////////////////////////////// Method ////////////////////////////////////////////////////////////////////////////////////////// Private //////////////////////////////////////////////////////////////////////////////// Event #region 실행 버튼 클릭시 처리하기 - runButton_Click(sender, e) /// <summary> /// 실행 버튼 클릭시 처리하기 /// </summary> /// <param name="sender">이벤트 발생자</param> /// <param name="e">이벤트 인자</param> private void runButton_Click(object sender, EventArgs e) { Button button = sender as Button; button.Enabled = false; Thread thread = new Thread(new ThreadStart(ProcessThread)); thread.Start(); } #endregion //////////////////////////////////////////////////////////////////////////////// Function #region 스레드 처리하기 - ProcessThread() /// <summary> /// 스레드 처리하기 /// </summary> private void ProcessThread() { for(int i = 0; i < 10; i++) { this.InvokeIfRequired((form) => { this.messageLabel.Text = (i + 1).ToString(); }); Thread.Sleep(500); } this.InvokeIfRequired ( (form) => { this.messageLabel.Text = string.Empty; this.runButton.Enabled = true; } ); } #endregion } } |
TestProject.zip