本来这个不应该成为重复的话题,但是花了好些功夫没在网上找到能一目了然的解决办法,
只好再啰嗦几句,也算给自己留条后路。
我个人理解一般函数的无类型指针参数就是传递用户自定义类型用,《GTK+2.0编程范例》
也讲到给g_signal_connect传递多个参数时可以定义包含多个参数的结构体,然后将这个结构
体的变量指针传递进去,但也仅仅说了这么多。我在实践的时候,程序却报错:
:GLib-GObject-WARNING **: invalid unclassed pointer in cast to 'GtkEntry'
:Gtk-CRITICAL **: file gtkentry.c: line 3508 (gtk_entry_get_text):assertion
'GTK_IS_ENTRY (entry)' failed
部分源代码如下:
/* file reg.c */
struct reg_text {
GtkEntry * user;
GtkEntry * sn;
};
void
on_reg_btnOk_clicked (GtkWidget * btn, gpointer data)
{
struct reg_text * pRt = (struct reg_text *)data;
const gchar * pUser = gtk_entry_get_text (GTK_ENTRY(pRt->user));
const gchar * pSn = gtk_entry_get_text (GTK_ENTRY(pRt->sn));
g_print ("Username: %s\n", pUser);
g_print ("SerialNumber: %s\n", pSn);
}
GtkWidget * CreateRegisterWindow()
{
........
struct reg_text rt;
GtkWidget * entUser;
GtkWidget * entSn;
........
rt.user = GTK_ENTRY(entUser);
........
rt.sn = GTK_ENTRY(entSn);
........
g_signal_connect (G_OBJECT(btnOk), "clicked",
G_CALLBACK (on_reg_btnOk_clicked), (gpointer)&gRt);
........
return 0;
}
然后就是查资料,花了一个多小时才找到一篇和这个问题沾边的帖子,页面原链接
看到楼主的问题解决很是高兴,他说:
“问题已解决,本来是传递结构变量的地址,后来改成指针就OK了,不过不明白为什么”
但我看了以后更不明白,变量的地址和指针本质上有什么不同么?
幸好最后的回复给了我提示,在此感谢那位老兄!他说:
“你不会是把一个local变量传进去了吧”
仔细想了想,我是在CreateRegisterWindow()函数中定义了 struct reg_text rt,
相对整个工程来说,struct reg_text rt就是个局部变量。我把 struct reg_text rt;
拿到了CreateRegisterWindow()函数外定义:static struct reg_text rt;
重新编译运行,一切ok!
和同事讨论了一下,这种事件处理的回调函数的生存周期和整个程序的生存周期一样,而且是
异步调用的。如果传递给它的参数是个局部变量的话,当这个回调函数被调用时那个变量非常
可能已经不存在了,所以只能传递给它一个和它具有同样生存周期的变量。(注:不知道“生存
周期”还是“作用域”合适,只是前者说起来好像更形象一些,易理解,权且用它。)