30초마다 백그라운드 스레드에서 GUI를 안전하게 업데이트하는 방법은 무엇입니까?
Jan 10, 2025 am 10:20 AM백그라운드 스레드에서 30초마다 GUI를 안전하게 업데이트하세요
이 문서에서는 애플리케이션에서 스레드를 안전하게 사용하고, 오류를 방지하고, 30초마다 백그라운드 스레드에서 GUI를 업데이트해야 하는 필요성을 구현하는 방법을 살펴봅니다.
사용자가 처음에 BackgroundWorker를 사용하여 while 루프에서 실행하려고 시도했지만 백그라운드 작업자 스레드가 STA 스레드가 아니기 때문에 예외가 발생했습니다.
올바른 접근 방식은 데이터베이스 호출과 GUI 업데이트를 분리하는 것입니다. 데이터베이스 호출은 백그라운드 스레드에서 수행되어야 합니다(BackgroundWorker 사용). 호출이 완료된 후 진행 이벤트가 트리거되어 메인 스레드에 GUI 업데이트를 알립니다.
이 접근 방식을 보여주는 사용자 기여 코드:
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) { // 大型数据库任务 } }
그러나 사용자는 30초마다 업데이트를 트리거하는 방법이라는 또 다른 과제에 직면합니다. RunWorkerCompleted
이벤트 핸들러에 10초의 절전 모드를 추가하면 GUI가 정지되므로 실행 가능하지 않습니다.
이 문제를 해결하기 위해 타이머를 사용하여 주기적으로 Update
메서드를 호출하면 데이터베이스 호출과 후속 GUI 업데이트가 시작됩니다. 더 나은 해결책은 RunWorkerCompleted
에서 Update()
을 재귀적으로 호출하는 것을 피하고 대신 System.Timers.Timer
또는 System.Windows.Forms.Timer
을 사용하여 업데이트 빈도를 제어하는 것입니다. 이렇게 하면 GUI가 응답성을 유지하고 잠재적인 스택 오버플로 오류를 방지할 수 있습니다.
제안되는 개선된 코드 구조는 다음과 같습니다(System.Timers.Timer
사용).
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) { // 大型数据库任务 } }
이 개선된 코드는 타이머를 사용하여 업데이트 빈도를 제어하고 재귀 호출로 인한 위험을 방지하며 BeginInvoke
을 사용하여 UI 업데이트가 UI 스레드에서 실행되도록 하여 GUI의 응답성을 보장합니다. 타이머를 중지하려면 적절한 시간에 _timer.Stop()
을 호출하는 것을 잊지 마세요.
위 내용은 30초마다 백그라운드 스레드에서 GUI를 안전하게 업데이트하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

인기 기사

인기 기사

뜨거운 기사 태그

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











C 언어 함수에 의해 반환 된 값 유형은 무엇입니까? 반환 값을 결정하는 것은 무엇입니까?

STL (정렬, 찾기, 변환 등)의 알고리즘을 효율적으로 사용하려면 어떻게합니까?
