GTK编程-给回调函数传递多个用户参数
2013-06-13 16:46
429 查看
本来这个不应该成为重复的话题,但是花了好些功夫没在网上找到能一目了然的解决办法,
只好再啰嗦几句,也算给自己留条后路。
我个人理解一般函数的无类型指针参数就是传递用户自定义类型用,《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!
和同事讨论了一下,这种事件处理的回调函数的生存周期和整个程序的生存周期一样,而且是
异步调用的。如果传递给它的参数是个局部变量的话,当这个回调函数被调用时那个变量非常
可能已经不存在了,所以只能传递给它一个和它具有同样生存周期的变量。(注:不知道“生存
周期”还是“作用域”合适,只是前者说起来好像更形象一些,易理解,权且用它。)
只好再啰嗦几句,也算给自己留条后路。
我个人理解一般函数的无类型指针参数就是传递用户自定义类型用,《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!
和同事讨论了一下,这种事件处理的回调函数的生存周期和整个程序的生存周期一样,而且是
异步调用的。如果传递给它的参数是个局部变量的话,当这个回调函数被调用时那个变量非常
可能已经不存在了,所以只能传递给它一个和它具有同样生存周期的变量。(注:不知道“生存
周期”还是“作用域”合适,只是前者说起来好像更形象一些,易理解,权且用它。)
相关文章推荐
- Java中用户向系统传递参数的三种基本方式
- Java中用户向系统传递参数的三种基本方式实例分享
- C#调用C/C++ DLL 参数传递和回调函数的总结
- 向用户控件传递参数的问题
- 另类参数对话框编程(回调函数,消息捕捉)
- 用户自定义类型(User-defined Type)参数的传递
- 函数名作为参数传递 与 回调函数
- SQL中函数返回表,传递一个参数,无法唤醒用户自定义函数,解决方法!
- JNI编程之如何传递参数(二)——数组类型的传递
- 传递参数(C# 编程指南)(http://msdn.microsoft.com/zh-cn/library/0f66670z(VS.80).aspx)
- linux shell编程指南第二十章------向脚本传递参数
- JavaScript浏览器编程之——使用URL传递参数
- ubuntu 内核模块编程+模块参数传递+模块符号表输出
- Python线程池编程和参数的传递
- Android网络编程之通过Post传递参数
- ASP.NET MVC编程入门--MVC5 传递参数与初始化数据
- Android jni 编程(参数的传递,成员,方法的)相互访问
- 向用户控件传递参数的问题
- ASP.NET温故而知新学习系列之ASP.NET多线程编程—.NET下的多线程编程利用thread.Start()传递参数(四)
- 确保线程结束之前,传递给线程回调函数的参数的有效性