2010-12-05 4 views
1

다음 예제는 here에서 가져온 것으로, "Increase - Decrease"제목 아래에 표시됩니다.Gtk + 콜백 함수 및 신호 도움

#include <gtk/gtk.h> 

gint count = 0; 
char buf[5]; 

void increase(GtkWidget *widget, gpointer label) 
{ 
    count++; 

    sprintf(buf, "%d", count); 
    gtk_label_set_text(label, buf); 
} 

void decrease(GtkWidget *widget, gpointer label) 
{ 
    count--; 

    sprintf(buf, "%d", count); 
    gtk_label_set_text(label, buf); 
} 

int main(int argc, char** argv) { 

    GtkWidget *label; 
    GtkWidget *window; 
    GtkWidget *frame; 
    GtkWidget *plus; 
    GtkWidget *minus; 

    gtk_init(&argc, &argv); 

    window = gtk_window_new(GTK_WINDOW_TOPLEVEL); 
    gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER); 
    gtk_window_set_default_size(GTK_WINDOW(window), 250, 180); 
    gtk_window_set_title(GTK_WINDOW(window), "+-"); 

    frame = gtk_fixed_new(); 
    gtk_container_add(GTK_CONTAINER(window), frame); 

    plus = gtk_button_new_with_label("+"); 
    gtk_widget_set_size_request(plus, 80, 35); 
    gtk_fixed_put(GTK_FIXED(frame), plus, 50, 20); 

    minus = gtk_button_new_with_label("-"); 
    gtk_widget_set_size_request(minus, 80, 35); 
    gtk_fixed_put(GTK_FIXED(frame), minus, 50, 80); 

    label = gtk_label_new("0"); 
    gtk_fixed_put(GTK_FIXED(frame), label, 190, 58); 

    gtk_widget_show_all(window); 

    g_signal_connect(window, "destroy", 
     G_CALLBACK (gtk_main_quit), NULL); 

    g_signal_connect(plus, "clicked", 
     G_CALLBACK(increase), label); 

    g_signal_connect(minus, "clicked", 
     G_CALLBACK(decrease), label); 

    gtk_main(); 

    return 0; 
} 

내가 궁금하고있는 무슨의 g_signal_connect(plus, "clicked",G_CALLBACK(increase), label); 함수는 인수가 void increase(GtkWidget *widget, gpointer label)을있는 기능 증가로 "라벨"을 보냅니다. 이제 증가 함수에서 gtk_label_set_text() 함수는 첫 번째 인수로 GtkLabel의 데이터 형식이 필요하지만 증가 함수에 대한 인수로 GtkWidget 변수와 void 포인터 label 만 볼 수 있습니다. 그렇다면 gtk_label_set_text()는 어떻게 작동합니까?

답변

3

C (그러나 C++)에서 void*을 다른 유형의 포인터로 암시 적으로 캐스트 할 수 있습니다.

당신의 코드는 단지 매개 변수 전달과 같은 일을하고
int *myIntArray = malloc(10 * sizeof(int)); // allocate array of 10 ints 

:

void gtk_label_set_text(GtkLabel *label, const char *text); 
void *label = ...; 
gtk_label_set_text(label, "some string"); // label is implicitly cast from 
              // void* to GtkLabel* 
+0

감사합니다, 또 다른 한가지는, 왜 우리가 포함 것 void* 반환, malloc으로 메모리를 할당 할 때 그것은 매우 일반적으로 볼 수 있어요 증가 함수의'GtkWidget * widget'? – silent

+0

@ sil3nt :'GtkWidget * widget'은 함수가 매우 특정한 프로토 타입을 가져야하기 때문에 함수에 포함됩니다. 'GtkButton :: clicked' 콜백은 콜백 함수가 그 시그니처를 가질 것을 요구하며 그것은 첫 번째 매개 변수로'GtkWidget *'을 전달합니다. 콜백에 잘못된 서명이있는 경우 잘못된 매개 변수가 표시되고 충돌 가능성이 큽니다. –

+0

증가 함수의 첫 번째 매개 변수는'g_signal_connect (plus, clicked ", G_CALLBACK (increase), label)를 통해 자동으로 전달됩니다; – silent