我有一个我正在尝试为 Windows 创建的小应用程序。我遇到了混合旨在处理某些数据的后台线程的问题。这个后台引擎需要更新应用程序 gui(Windows 窗体)并从中获取信息。
这是基本的应用程序主要。
int main() {
Engine engine;
Gui g;
engine.run(); // creates a new thread for engine logic
g.ShowDialog();
bool running = false;
while(1)
{
// Update gui with information from the engine
g.update(engine.GetState());
// transition to running
if(g.isRunning() && !running)
{
engine.play();
running = true;
}
// transition to stopped
else if(!g.isRunning() && running)
{
engine.stop();
running = false;
}
}
}
我的主要问题来自 Gui 类是管理的。请参阅下面的类声明。
public ref class Gui : public System::Windows::Forms::Form
我真的不能把这两个东西混合起来,起初我只想把引擎扔到 Gui 中,但这不起作用,因为它是不受管理的。
您会注意到这里的问题是调用 ShowDialog(),因为这会使对话框成为模态,之后不会执行任何代码。但是,如果我使用 Show() ... Gui 根本不会更新或处理任何输入。
解决方案:
我在 Gui 类中创建了一个后台工作程序,因此引擎包含在 Gui 中,但在另一个线程上运行。
void InitializeBackgoundWorker()
{
this->backgroundWorker1 = gcnew System::ComponentModel::BackgroundWorker;
backgroundWorker1->DoWork += gcnew DoWorkEventHandler( this, &Gui::backgroundWorker1_DoWork );
backgroundWorker1->RunWorkerAsync( );
}
delegate void UpdateCallback(int hp, int maxhp);
void UpdateGui(int hp, int maxhp)
{
this->playerHealthBar->Value = ((float)(hp)/(float)(maxhp) * 100.0f);
};
void backgroundWorker1_DoWork( Object^ sender, DoWorkEventArgs^ e )
{
aBotEngine engine;
while(true)
{
engine.runnable(NULL);
array<Object^>^args = gcnew array<Object^>(2);
args[0] = engine.getPlayerHp();
args[1] = engine.getPlayerMaxHp();
this->playerHealthBar->Invoke(gcnew UpdateCallback(this, &Gui::UpdateGui), args);
}
};