如何避免阻塞函数g_application_run

时间:2017-10-24 12:00:42

标签: gtk

我有一个具有以下模式的应用程序(它是一个嵌入式应用程序),我不想改变它:

void
main(int argc, char *argv[])
{
  some_init_functions();
  while(1) {
    some_functions();
  }
}

我想在Windows上模拟应用程序,所以我想使用Gtk + 3来显示一个模拟原始应用程序中使用的显示的窗口。

问题是阻塞的函数g_application_run()。

是否可以手动调用Gtk + / GLib函数来处理挂起事件并立即返回?我会在while(1)循环中调用此函数。

我读到了关于gtk_main_iteration_do()的信息,但我不知道如何在不调用g_application_run()的情况下使用它。

2 个答案:

答案 0 :(得分:1)

将您的应用程序集成到GUI中是一项工作,而不是相反。您应该拨打g_application_run,然后添加一个g_timeout_addg_idle_add的来自您回叫的活动来源。在该回调中,只需拨打some_functions()一次即可。 GTK主循环将一次又一次地调用它。

答案 1 :(得分:0)

有可能跳出 g_application_run 循环。事件可以由 g_main_context_iteration(GMainContext *context, gboolean may_block) 处理。使用 may_block = TRUE,它只会在处理完事件后返回主循环。使用 may_block = FALSE,只处理最高优先级的事件,之后控制返回到循环。

下面的代码基于 GTK Reference Manual 中的 example-0.c 代码,但使 g_application_run 中的流程变得明确。请注意,不处理命令行参数,并且省略了几项检查。

添加超时事件,与 may_block = TRUE 结合,主循环至少每秒前进一次。此外,在 destroy 事件上添加了一个回调,这样,结合全局变量,如果窗口关闭,主循环可以停止。

作为对所问问题的直接回答:您需要确保执行正确的设置和分解,并将 g_main_context_iteration (context, may_block); 添加到您的 while (1) 循环中。

#include <gtk/gtk.h>

static bool application_running;  // As use_count is private

// Add close_window event watcher to tell our application loop we should stop
static void
close_window (void)
{
  printf("Window is closed\n");
  application_running = FALSE;
}

// Add timeout to make sure the application loop is unblocked once per second
static gboolean
timeout (gpointer data)
{
  printf("timeout\n");
  return true;
}

static void
activate (GtkApplication* app,
          gpointer        user_data)
{
  GtkWidget *window;

  window = gtk_application_window_new (app);

  // Add destroy call-back so we know when window is closed.
  g_signal_connect (window, "destroy", G_CALLBACK (close_window), NULL);

  gtk_window_set_title (GTK_WINDOW (window), "Window");
  gtk_window_set_default_size (GTK_WINDOW (window), 200, 200);
  gtk_widget_show (window);
}

int
main (int    argc,
      char **argv)
{
  GtkApplication *app;
  int status;

  app = gtk_application_new ("org.gtk.example", G_APPLICATION_FLAGS_NONE);
  g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);

  g_timeout_add_seconds (1, timeout, NULL);

  // **** begin g_application_run alternative ****
  // Setup
  GMainContext *context;
  gboolean acquired_context;

  context = g_main_context_default ();
  acquired_context = g_main_context_acquire (context);
  g_return_val_if_fail (acquired_context, 0);

  GError *error = NULL;
  if (!g_application_register (G_APPLICATION (app), NULL, &error))
    {
      g_printerr ("Failed to register: %s\n", error->message);
      g_error_free (error);
      return 1;
    }

  g_application_activate (G_APPLICATION (app));

  // Main event loop
  application_running = true;
  int loop = 0;
  while (application_running) {
      bool may_block = TRUE;
      // may_block = TRUE: g_main_context_iteration blocks execution if no events are coming
      // may_block = FALSE: g_main_context_iterations processes pending events and continues
      g_main_context_iteration (context, may_block);
      printf("Loop %i\n", loop++);
    }

  // Release
  g_settings_sync ();
  while (g_main_context_iteration (context, FALSE)) 
    ;
  g_main_context_release (context);
  // **** end g_application_run alternative ****

  g_object_unref (app);

  return status;
}