백그라운드 스레드에서 30초마다 GUI를 안전하게 업데이트하세요
이 문서에서는 애플리케이션에서 스레드를 안전하게 사용하고, 오류를 방지하고, 30초마다 백그라운드 스레드에서 GUI를 업데이트해야 하는 필요성을 구현하는 방법을 살펴봅니다.
사용자가 처음에 BackgroundWorker를 사용하여 while 루프에서 실행하려고 시도했지만 백그라운드 작업자 스레드가 STA 스레드가 아니기 때문에 예외가 발생했습니다.
올바른 접근 방식은 데이터베이스 호출과 GUI 업데이트를 분리하는 것입니다. 데이터베이스 호출은 백그라운드 스레드에서 수행되어야 합니다(BackgroundWorker 사용). 호출이 완료된 후 진행 이벤트가 트리거되어 메인 스레드에 GUI 업데이트를 알립니다.
이 접근 방식을 보여주는 사용자 기여 코드:
<code class="language-C#">public class UpdateController { private UserController _userController; private BackgroundWorker _backgroundWorker; // ... public void Update() { _backgroundWorker.RunWorkerAsync(); } void backgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e) { // UI 更新 _userController.UpdateUsersOnMap(); Update(); // 此处存在潜在问题,递归调用可能导致堆栈溢出 } public void backgroundWorker_DoWork(object sender, DoWorkEventArgs e) { // 大型数据库任务 } }</code>
그러나 사용자는 30초마다 업데이트를 트리거하는 방법이라는 또 다른 과제에 직면합니다. RunWorkerCompleted
이벤트 핸들러에 10초의 절전 모드를 추가하면 GUI가 정지되므로 실행 가능하지 않습니다.
이 문제를 해결하기 위해 타이머를 사용하여 주기적으로 Update
메서드를 호출하면 데이터베이스 호출과 후속 GUI 업데이트가 시작됩니다. 더 나은 해결책은 RunWorkerCompleted
에서 Update()
을 재귀적으로 호출하는 것을 피하고 대신 System.Timers.Timer
또는 System.Windows.Forms.Timer
을 사용하여 업데이트 빈도를 제어하는 것입니다. 이렇게 하면 GUI가 응답성을 유지하고 잠재적인 스택 오버플로 오류를 방지할 수 있습니다.
제안되는 개선된 코드 구조는 다음과 같습니다(System.Timers.Timer
사용).
<code class="language-C#">public class UpdateController { private UserController _userController; private BackgroundWorker _backgroundWorker; private System.Timers.Timer _timer; public UpdateController() { _backgroundWorker = new BackgroundWorker(); _backgroundWorker.DoWork += backgroundWorker_DoWork; _backgroundWorker.RunWorkerCompleted += backgroundWorker_RunWorkerCompleted; _timer = new System.Timers.Timer(30000); // 30秒 _timer.Elapsed += Timer_Elapsed; _timer.AutoReset = true; } public void Start() { _timer.Start(); } private void Timer_Elapsed(object sender, System.Timers.ElapsedEventArgs e) { _backgroundWorker.RunWorkerAsync(); } void backgroundWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e) { // UI 更新,确保在UI线程执行 _userController.BeginInvoke(new Action(() => _userController.UpdateUsersOnMap())); } public void backgroundWorker_DoWork(object sender, DoWorkEventArgs e) { // 大型数据库任务 } }</code>
이 개선된 코드는 타이머를 사용하여 업데이트 빈도를 제어하고 재귀 호출로 인한 위험을 방지하며 BeginInvoke
을 사용하여 UI 업데이트가 UI 스레드에서 실행되도록 하여 GUI의 응답성을 보장합니다. 타이머를 중지하려면 적절한 시간에 _timer.Stop()
을 호출하는 것을 잊지 마세요.
위 내용은 30초마다 백그라운드 스레드에서 GUI를 안전하게 업데이트하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!