为了测试我对Gtk其他位的理解,我想编写一个程序,该程序始终有一个事件可供主循环使用。我写了这个简短的程序来尝试这样做:

#include <gtk/gtk.h>

static void toggle(GtkWidget *check, gpointer data)
{
    gboolean checked;
    g_object_get(check, "active", &checked, NULL);
    g_object_set(check, "active", !checked, NULL);
}

int main(int argc, char *argv[])
{
    GtkWidget *window, *check;
    gtk_init(&argc, &argv);
    window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
    check  = gtk_check_button_new();
    g_signal_connect(check, "toggled", G_CALLBACK(toggle), NULL);
    gtk_container_add(GTK_CONTAINER(window), check);
    gtk_widget_show_all(window);
    gtk_main();
}

当我运行该程序并单击复选框时,它会出现段错误。是什么赋予了?保持主循环繁忙的正确方法是什么?

(旁注:在进行段错误之前,它可靠地切换了2048次-一个可疑的整数。)

最佳答案

toggle处理程序中,您正在设置checked,这将导致toggle信号被发出,从而重新调用处理程序...

#11564 0xb775ba50 in g_closure_invoke () from /usr/lib/libgobject-2.0.so.0
#11565 0xb776e5d0 in ?? () from /usr/lib/libgobject-2.0.so.0
#11566 0xb77774d6 in g_signal_emit_valist () from /usr/lib/libgobject-2.0.so.0
#11567 0xb7777682 in g_signal_emit () from /usr/lib/libgobject-2.0.so.0
#11568 0xb7e067ba in gtk_toggle_button_toggled ()

我并没有一直跟踪,但是我可以看到> 11000帧如何导致段错误。

回答另一个问题:我认为保持主循环完整的方法是使用g_idle_add()调用:
#include <gtk/gtk.h>

static void toggle(GtkWidget *check, gpointer data)
{
  g_print(".");
}

GtkWidget *window, *check;

static gboolean
toggle_it()
{
  gboolean checked;
  g_object_get(check, "active", &checked, NULL);
  g_object_set(check, "active", !checked, NULL);
  return TRUE;
}

int main(int argc, char *argv[])
{
    gtk_init(&argc, &argv);
    window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
    check  = gtk_check_button_new();
    g_signal_connect(check, "toggled", G_CALLBACK(toggle), NULL);
    gtk_container_add(GTK_CONTAINER(window), check);
    gtk_widget_show_all(window);
    g_idle_add((GSourceFunc)toggle_it, NULL);
    gtk_main();
}

关于c - 试图保持主循环繁忙时出现段错误,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/10857773/

10-11 21:16