int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine) (void *), void *arg);
//用于创建线程
//第一个参数:一个用户自定的指向类型为pthread_t的指针,头文件中pthread_t类型其实为unsigned long型。这个指针指向的内存单元将会在创建完线程以后,会变为线程ID
//第二个参数:是线程的属性,暂时没用到,直接填NULL
//第三个参数:线程所对应的函数,也就是创建新的线程后,该线程要执行的内容。
//第四个参数:第三个参数中所填函数的参数,当参数大于1时,需要创建一个结构体,把结构体地址填在这里。
上面简单介绍了一下这个函数的参数大概时什么个东西,但我刚接触这个函数的时候真的是一脸懵,看视频也是说的模模糊糊的,然后看资料也是个个都讲得差不多,很难懂,文邹邹的,然后只能通过自己对视频和资料的大概理解,再来做实验证明自己的理解是不是对的。
其实主要不懂的地方只是第一个参数,不知道怎么就变成线程ID了,到底是怎么来的这个线程ID。
下面给出我的思路:
看资料上说,当 pthread_create成功返回时,由tidp指向的内存单元被设置为新创建线程的线程ID。那么,我假设变量pthread_t t1就是这第一个参数,当成功返回时,t1的值是多少?是t1这个变量的地址?还是t1的初始值?还是说这个值是由pthread_create赋予的?
带着这些疑问,我开始了我的实践。
代码如下:
#include <stdio.h>
#include <pthread.h>
void *fun1(void *arg) //新建线程所对应的函数
{
printf("------the following content is from fun1------\n"); //分隔符
printf("val=%d\n",*((int *)arg)); //打印传进来的参数看是否正确,有没有出错
printf("fun1 thread id:%ld\n",(unsigned long)pthread_self()); //看新建的线程的线程ID是多少
}
int main()
{
pthread_t t1 = 1; //给t1赋予初值,以便观察是否被改变
int val = 10; //给参数赋予初值
printf("address of t1 in Hex:%p\n",&t1); //打印变量t1的地址(十六进制)
printf("address of t1 in Dec:%ld\n",(unsigned long)(&t1)); //打印变量t1的地址(十进制)
printf("before pthread_create t1=%ld\n",(unsigned long)t1); //打印t1在创建新线程之前的值
pthread_create(&t1,NULL,fun1,(void *)&val); //创建新线程
printf("after pthread_create t1=%ld\n",(unsigned long)t1); //打印t1在创建新线程以后的值
printf("main thread id:%ld\n",(unsigned long)pthread_self()); //打印main函数这个线程的ID
while(1); //保持进程不退出,如果进程消失了,线程也就消失了,就看不到线程执行的结果了
}
下面是运行结果:
从运行结果中我们可以看出,在创建新线程后,变量t1的值就会被改变,而且在使用的时候,函数要求第一个参数是指针,所以我们填写的是t1的地址,但创建新线程后的线程ID跟t1地址,这两个值并不一样,所以说,并不是把t1地址的值作为新线程的ID
然后再来看t1在创建新线程之前的值和创建新线程之后的值,很明显完全不一样,所以也不是把t1的初始值作为新线程的ID
最后,得出的结论,这线程ID就是pthread_create赋予的,但和t1的地址,值都没有关系,要求参数是指针可能也只是函数方便进行操作罢了。