- 论坛徽章:
- 0
|
我这还有稍微具体的介绍
在glib中提供了一个对于线程进行了封装,在国力把中有 Threads 的api。
在编译含有thread的程序的时候,在Makefile中需要使用
`pkg-config --cflags --libs gtk+-2.0 gthread-2.0`
这是因为thread在glib中是可选编译的,如果编译完成之后就会定义 G_THREADS_ENABLED ,
如果在编写线程可选的程序中可以检查这个宏是否定义。
在程序开始的时候,我们需要g_thread_supported(),然后使用g_thread_init(NULL),
在使用任何有关gtk的函数的时候,我们都需要使用gdk_threads_enter()和gdk_threads_leave()
这两个成对的函数。
下面是一个实例:
#include <gtk/gtk.h>
/*创建的线程: 更改标签文本*/
void thread1(GtkWidget *label){
int i;
char str[3];
for(i=1;i<=10;i++){
sleep(1);
sprintf(str,"%d",i);
gdk_threads_enter();
gtk_label_set_text(GTK_LABEL(label),str);
gdk_threads_leave();
}
}
int main(int argc, char *argv[]){
GtkWidget *window;
GtkWidget *vbox, *label;
if(!g_thread_supported()) g_thread_init(NULL);
gdk_threads_init();
gtk_init(&argc,&argv);
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_title(GTK_WINDOW(window),"thread test");
g_signal_connect(G_OBJECT(window),"delete_event",
G_CALLBACK(gtk_main_quit),NULL);
gtk_container_set_border_width(GTK_CONTAINER(window),2);
vbox = gtk_vbox_new(FALSE,0);
gtk_container_add(GTK_CONTAINER(window),vbox);
label = gtk_label_new("num");
gtk_widget_set_size_request (label,50,50);
gtk_box_pack_start(GTK_BOX(vbox),label,FALSE,FALSE,5);
gtk_widget_show_all(window);
g_thread_create(thread1, label, FALSE, NULL);
gdk_threads_enter();
gtk_main();
gdk_threads_leave();
return FALSE;
}
在编写多线程的程序中,要注意的是全局变量的同步,这里介绍的是使用互斥锁的方法。
gthread中也提供了这方面的封装,
GMutex;
GMutex* g_mutex_new ();
void g_mutex_lock (GMutex *mutex);
gboolean g_mutex_trylock (GMutex *mutex);
void g_mutex_unlock (GMutex *mutex);
void g_mutex_free (GMutex *mutex);
一般而言,在主线程中对互斥锁进行 new 和 free。
在其他线程中使用,
if(g_mutex_trylock(mutex) == FALSE)
{
临界区操作,尽量不要太复杂。
g_mutex_unlock(mutex);
}
如果有什么错误和补充的地方,欢迎讨论。 |
|