pthread_once使用 -- 仅初始化一次

1、函数原型:

int pthread_once(pthread_once_t *once_control, void (*init_routine)(void));

pthread_once的作用为,在给定once_control的情况下,在整个程序中仅调用一次init_routine(在多线程中具体哪一个线程执行不一定)。如果再次调用,pthread_once将不会调用init_routine。

该函数执行成功返回0,执行失败返回错误码。

具体可以参考下如下man手册:

  • The first call to pthread_once() by any thread in a process, with a given once_control, will call the init_routine() with no arguments.
  • Subsequent calls to pthread_once() with the same once_control will not call the init_routine(). On return from pthread_once(), it is guaranteed that init_routine() has completed. The once_control parameter is used to determine whether the associated initialization routine has been called.

以下说明了once_control需要初始化,如果初值不是PTHREAD_ONCE_INIT,pthread_once行为就不正常:

  • The function pthread_once() is not a cancellation point. However, if init_routine() is a cancellation point and is cancelled, the effect on once_control is as if pthread_once() was never called.
  • The constant PTHREAD_ONCE_INIT is defined by header <pthread.h>.
  • The behavior of pthread_once() is undefined if once_control has automatic
    storage duration or is not initialized by PTHREAD_ONCE_INIT.

2、pthread_once_t、PTHREAD_ONCE_INIT定义

pthread_once_t变量的定义:

typedef __darwin_pthread_once_t pthread_once_t;

typedef struct _opaque_pthread_once_t __darwin_pthread_once_t;

struct _opaque_pthread_once_t {
	long __sig;
	char __opaque[__PTHREAD_ONCE_SIZE__];
};

#define __PTHREAD_ONCE_SIZE__		8

PTHREAD_ONCE_INIT的定义

#define PTHREAD_ONCE_INIT {_PTHREAD_ONCE_SIG_init, {0}}

#define _PTHREAD_ONCE_SIG_init		0x30B1BCBA

3、代码实例:

/* donald: pthread_once中的thread_init只会执行一次,在多线程中具体哪一个线程执行不一定
  DESCRIPTION:  一次性初始化
    int pthread_once(pthread_once_t* once_control, void (*init_routine)(void));
    如果once_control为0,init_routine()就会执行
    pthread_once()成功返回之后,once_control会变为2
 */
#include <iostream>
#include <pthread.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <signal.h>
#include <errno.h>

pthread_once_t once = PTHREAD_ONCE_INIT;
pthread_t tid;

void thread_init()
{
    printf("init::I'm in thread 0x%x\n", tid);
}


void *thread_fun1(void *arg)
{
    tid = pthread_self();
    printf("th1:I'm thread 0x%x\n", tid);
    printf("th1:once is %d\n", once);
    pthread_once(&once, thread_init);
    printf("th1:once is %d\n", once);

    return NULL;
}

void *thread_fun2(void *arg)
{
    // sleep(2);
    tid = pthread_self();
    printf("th2:I'm thread 0x%x\n", tid);
    pthread_once(&once, thread_init);
    printf("th2:once is %d\n", once);
    printf("th2:once is %d\n", once);
    return NULL;
}

int main()
{
    pthread_t tid1, tid2;
    int err;
    printf("main:once is %lu\n", once);
    // std::cout << "main:once is " << once << std::endl;

    err = pthread_create(&tid1, NULL, thread_fun1, NULL);
    if(err != 0)
    {
        printf("create new thread 1 failed\n");
        return 0;
    }
    err = pthread_create(&tid2, NULL, thread_fun2, NULL);
    if(err != 0)
    {
        printf("create new thread 1 failed\n");
        return 0;
    }

    printf("I'm main thread 0x%x, tid1:0x%x, tid2:0x%x\n", pthread_self(), tid1, tid2);

    pthread_join(tid1, NULL);
    pthread_join(tid2, NULL);

    return 0;
}

运行结果:

main:once is 816954554
I'm main thread 0xb28ddc0, tid1:0xf8c3000, tid2:0xf946000
th1:I'm thread 0xf8c3000
th1:once is 816954554
th2:I'm thread 0xf946000
init::I'm in thread 0xf946000
th1:once is 1330529093
th2:once is 1330529093
th2:once is 1330529093

结果释义:

  • 在main中输出once,值为816954554,816954554实际上是0x30B1BCBA(16进制),根据上面PTHREAD_ONCE_INIT的定义,可以知道这里0x30B1BCBA就是其初始值。

  • 根据输出的内容可以看出,thread_init只运行了一次(输出init::I’m in thread 0xf946000),之后的once值也未改变。

4、应用 & 参考

**应用:**可以用pthread_once来实现一个单例模式,具体实现见另一篇博文【单例模式实现方法】

参考:

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值