关于g_signal_connect,g_cclosure_new,gpointer的问题

时间:2011-01-23 16:13:11

标签: c gtk gobject

// gcc 1.c -o 0 $(pkg-config --cflags --libs gtk+-2.0)
#include <gtk/gtk.h>
#include <gdk/gdkkeysyms.h>
struct tst
{
    GtkWidget *win, *vb, *ent, *btn, *lbl;
    GtkAccelGroup *acc;
    GClosure *cls;
};
static void print_val(int nmb)
{
    g_printf("%d\n", nmb);
}
static void window_new()
{
    struct tst *prg = g_new0(struct tst, 1);
    prg->win = gtk_window_new(GTK_WINDOW_TOPLEVEL);
    prg->vb = gtk_vbox_new(TRUE, 0);
    prg->ent = gtk_entry_new();
    prg->btn = gtk_button_new_with_label("Print!");
    prg->lbl = gtk_label_new("Enter the string.");
    prg->acc = gtk_accel_group_new();
    int nmb = 140;
    prg->cls = g_cclosure_new(G_CALLBACK(print_val), nmb, NULL);
    gtk_container_add(GTK_CONTAINER(prg->win), GTK_WIDGET(prg->vb));
    gtk_box_pack_start(GTK_BOX(prg->vb), GTK_WIDGET(prg->ent), FALSE, FALSE, 0);
    gtk_box_pack_start(GTK_BOX(prg->vb), GTK_WIDGET(prg->btn), FALSE, FALSE, 0);
    gtk_box_pack_start(GTK_BOX(prg->vb), GTK_WIDGET(prg->lbl), FALSE, FALSE, 0);
    g_signal_connect(prg->win, "destroy", G_CALLBACK(gtk_main_quit), NULL);
    g_signal_connect(prg->btn, "clicked", G_CALLBACK(print_val), (gpointer)nmb);
    gtk_window_set_title(GTK_WINDOW(prg->win), "Enter the string");
    gtk_window_add_accel_group(GTK_WINDOW(prg->win), prg->acc);
    gtk_accel_group_connect(prg->acc, GDK_KEY_P, (GDK_CONTROL_MASK | GDK_SHIFT_MASK), GTK_ACCEL_MASK, prg->cls);
    gtk_widget_show_all(GTK_WIDGET(prg->win));
}
int main(int argc, char *argv[])
{
    gtk_init(&argc, &argv);
    window_new();
    gtk_main();
    return 0;
}

每次按下按钮或按Ctrl + Shift + P,我都想让这个程序在控制台上打印140.

但是当我按下按钮或Ctrl + Shirt + P时,这个程序打印出奇怪的值(不是140)。

我该怎么办?

2 个答案:

答案 0 :(得分:3)

您的print_val回调需要具有正确数量的参数。如果您在GTK文档中查找GtkButton::clicked信号,您将看到它具有此签名:

void user_function (GtkButton *button, gpointer user_data)

因此,单击按钮时传递给print_val的第一个参数是按钮本身的地址。这是您正在打印的“奇怪价值”。

与Jens的回答相反,GLib确实提供了一种将int强制转换为指针的可移植方式。所以为了实现你想要的,你需要像这样连接你的信号:

g_signal_connect(prg->btn, "clicked", G_CALLBACK(print_val), GINT_TO_POINTER(nmb));

然后像这样进行回调:

static void print_val(GtkButton *button, gpointer nmb)
{
    g_printf("%d\n", GPOINTER_TO_INT(nmb));
}

答案 1 :(得分:0)

我对gtk编程知之甚少,但我想这应该是

(gpointer)&nmb

无论如何,将int投射到指针上都不是一个好主意。