pthread_self()函数返回的是“真正”的线程 ID 吗?——gettid 与 pthread_self 的简单辨析

线程ID获取的两种方法:
1)gettid()函数
2)直接调用pthread_self()
我们分别用两个函数求父子主进程和其所属进程 id
先看一下代码,我们可以先预期一下输出结果

#include<stdio.h>
#include<pthread.h>
#include<unistd.h>

//由于pthread用户库内没有gettid()所以需要这个头文件以及下面的函数定义
#include<sys/syscall.h>
pid_t gettid()
{
     return syscall(SYS_gettid);
}

void *f(void *arg) {

    if(*(int *)arg){
        printf("tid 父进程所属线程 ID 为 %d\n",gettid());
        printf("pid 父进程所属线程 ID 为 %ld\n",pthread_self());
    }
    else{
        printf("tid 子进程所属线程 ID 为 %d\n",gettid());
        printf("pid 子进程所属线程 ID 为 %ld\n",pthread_self());
    }
    pthread_exit(NULL);

}
int main(){
    pid_t pid1;
    pthread_t tid1;
    pthread_t tid2;
    int *retval_1;
    int *retval_2;
    int a;
    pid1 = fork();
    
    if(pid1 == 0){
        a = 0;
        printf("pid 子进程主线程 ID 为 %ld\n",pthread_self());
        printf("tid 子进程主线程 ID 为 %d\n",gettid());
        pthread_create(&tid1,NULL,f,(void *)&a);
    }
    else{
        a = 1;
        printf("pid 父进程主线程 ID 为 %ld\n",pthread_self());
        printf("tid 父进程主线程 ID 为 %d\n",gettid());
        pthread_create(&tid2,NULL,f,(void *)&a);
    }
    pthread_join(tid1,(void*)retval_1);
    pthread_join(tid2,(void*)retval_2);
}

输出结果:

pid 父进程主线程 ID 为 140257979672384
tid 父进程主线程 ID 为 22558
pid 子进程主线程 ID 为 140257979672384
tid 子进程主线程 ID 为 22559
tid 父进程所属线程 ID 为 22560
pid 父进程所属线程 ID 为 140257971205888
tid 子进程所属线程 ID 为 22561
pid 子进程所属线程 ID 为 140257971205888

我们可以看到用 pthread_self() 函数得到的父子进程的主进程都分别和自己的所属线程 id 相等,但主进程 id 怎么能和所属线程 id 相等呢?

pthread_self 是posix描述的线程ID(并非内核真正的线程id),相对于进程中各个线程之间的标识号,对于这个进程内是唯一的,而不同进程中,每个线程的 pthread_self() 可能返回是一样的。
而 gettid 获取的才是内核中线程ID

posix是POSIX的线程标准,定义了创建和操纵线程的一套API,也就是我们现在用的API

下面来自其他博客的解答,感谢博主!
https://blog.csdn.net/rsyp2008/article/details/45150621

为什么需要两个不同的ID呢?
因为线程库实际上由两部分组成:内核的线程支持+用户态的库支持(glibc)
Linux在早期内核不支持线程的时候glibc就在库中(用户态)以纤程(就是用户态线程)的方式支持多线程了,
POSIX thread只要求了用户编程的调用接口对内核接口没有要求。

  • 19
    点赞
  • 34
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
pthread_create函数用于创建一个新的线程,其函数原型为: ```c int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine) (void *), void *arg); ``` 其中,参数thread是一个指向pthread_t类型的指针,用于存储新线程ID;参数attr是一个指向pthread_attr_t类型的指针,用于设置线程的属性,一般为NULL;参数start_routine是一个指向函数的指针,该函数将作为新线程的入口点;参数arg是一个指向void类型的指针,用于传递给start_routine函数的参数。 pthread_self函数用于获取当前线程ID,其函数原型为: ```c pthread_t pthread_self(void); ``` 下面是一个示例程序,演示了pthread_create函数和pthread_self函数的用法和程序的执行顺序: ```c #include <stdio.h> #include <pthread.h> void *thread_func(void *arg) { pthread_t tid = pthread_self(); printf("New thread created with ID %lu\n", tid); return NULL; } int main() { pthread_t tid; printf("Main thread ID is %lu\n", pthread_self()); pthread_create(&tid, NULL, thread_func, NULL); pthread_join(tid, NULL); return 0; } ``` 程序首先输出了主线程ID,然后调用pthread_create函数创建了一个新线程,并将其ID存储在tid变量中。pthread_create函数的第三个参数是一个指向函数的指针,该函数将作为新线程的入口点,这里我们传递了thread_func函数的地址。thread_func函数中调用了pthread_self函数获取当前线程ID,并输出到控制台。最后,主线程调用pthread_join函数等待新线程结束。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值