1. 入口函数:
主线程的入口函数一般可以认为是main函数;
一般线程的入口函数定义: void * thread_fun( void * arg);
2. 如何启动普通线程:
#include <pthread.h> |
int pthread_create(pthread_t *restrict_tidp, const pthread _attr_t *restrict_attr, void *(*start_rtn)(void), void *restrict_arg); |
1)成功返回0,否则返回错误号;
2)成功时:restrict_tidp是新线程id号;可以通过pthread_self()函数来得到此线程的id;
3)restrict_attr:新线程的属性,若为NULL,则默认属性;
4)start_rtn: 入口函数
5)restrict_arg: 入口函数的入参
3. 如何终止普通线程:
终止方式:
1)线程从启动例程中返回,返回值即线程的退出码,即return语句;
2)线程可以被同一进程中的其他线程取消;
3)线程调用pthread_exit退出;
#include <pthread.h> |
void pthread_exit(void *rval_ptr); |
注意:
#include <pthread.h> |
int pthread_cancel(pthread_t tid); |
1)成功返回0,否则返回错误码;
2)默认情况下,pthread_cancel函数会使tid线程表现等价于自己调用了pthread_exit(PTHREAD_CANCEL);
但线程可以选择忽略取消方式和控制取消方式;
3)此函数不是阻塞函数,并不等待线程终止,只是提出请求。
4. 如何获得线程的退出码?
#include <pthread.h> |
int pthread_join(pthread_t thread,void **rval_ptr); |
1)成功返回0,否则返回错误码
2)等待线程id等于thread的退出码,退出码放入rval_ptr中;
3)此函数只能捕获线程return/pthread_exit退出的;
4)此函数是阻塞函数,将一直等待直到thread线程return/pthread_exit或被其他线程取消
5.线程清理处理函数
#include <pthread.h> |
void pthread_cleanup _push(void (*rtn)(void *),void *arg); void pthread_cleanup _pop(int execute); |
demo:
void cleanup(void *arg)
{
printf(“cleanup: %s\n”,(char *)arg);
}
void *thr_fn(void *arg) /*线程入口地址*/
{
printf(“threadstart\n”);
pthread_cleanup_push(cleanup,”threadfirst handler”);/*设置第一个线程处理程序*/
pthread_cleanup_push(cleanup,”thread secondhandler”); /*设置第二个线程处理程序*/
printf(“threadpush complete\n”);
pthread_cleanup_pop(0);/*取消第一个线程处理程序*/
pthread_cleanup_pop(0);/*取消第二个线程处理程序*/
}
注意:1)为何需要:主要是清理线程所占用的资源,特别是锁资源;
2)注册顺序和执行顺序正好相反(原因:清理处理函数记录在栈中);
3)线程从启动例程return返回时,清理函数不执行,pthread_exit返回时,清理函数执行;
6. 一次性初始化函数
#include <pthread.h> |
pthread_once_t once_control=PTHREAD_ONCE_INIT; int pthread_once(pthread_once_t *once_control, void(*init_routine)(void)); |
once_control 控制变量 init_routine 初始化函数 |
1)正确返回0,否则返回错误码;
2)多线程情况下,一次性初始化比较麻烦,故posix提供了一次性初始化函数;
3)执行流程:线程首先检测控制变量,若已初始化完成,则立即返回;否则调用初始化函数,成功后记录初始化完成;
若其他线程正在执行初始化初始化函数,则调用线程等待,直到初始化函数完成。
7. 私有数据
#include <pthread.h> |
int pthread_key_create(pthread_key *key,void(*destructor)( void *)); |
int pthread_setspecific(pthread_key_t key, const void*value); void *pthread_getspecific(pthread_key_t key); |
1) 成功返回0; 否则返回错误码;
2)进程内所有线程共享进程地址空间,任何声明为静态/全局/堆变量,都可以被进程内所有的线程所读写,
所以有时线程需要私有数据,故posix提供了一种方法,创建线程键;
3)第一个参数为指向一个键值的指针,第二个参数指明了一个destructor函数(清理函数),如果这个参数不为空,那么当每个线程结束时,系统将调用这个函数来释放绑定在这个键上的内存块。这个函数常和函数pthread_once一起使用,为了让这个键只被创建一次。函数pthread_once声明一个初始化函数,第一次调用pthread_once时它执行这个函数,以后的调用将被它忽略。