线程控制原语之pthread_self和pthread_create函数

注意:使用线程库函数用gcc编译时,要加参数:-lpthreadlibpthread.so,因为线程库函数属于第三方c库函数,不是标准库函数(/lib、/usr/lib或者/usr/local/lib)。

1pthread_self函数

获取线程ID。其作用对应进程中 getpid() 函数。

pthread_t pthread_self(void);   返回值:成功:调用该函数的线程ID;失败:永远不会失败。

pthread_t:typedef  unsigned long int  pthread_t;  输出格式格式为:%lu

线程ID是进程内部,识别标志。(两个进程间,线程ID允许相同)

注意:不应使用全局变量 pthread_t tid(因为全局变量被一个进程中的多个线程共享),在子线程中通过pthread_create传出参数来获取线程ID或者使用pthread_self。

2pthread_create函数

创建一个新线程。其作用对应进程中fork() 函数。

int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine) (void *), void *arg);

返回值:成功:0;失败:错误编号。Linux环境下,所有线程特点,失败均直接返回错误编号,即直接返回errno number。

参数1:传出参数,新建线程的线程ID

参数2:传入参数,通常传NULL,表示使用线程默认属性。若想使用具体属性也可以修改该参数。

参数3:函数指针,指向线程主控函数(线程体),该函数运行结束,则线程结束。

参数4:线程主函数执行期间所使用的参数,若不同参数传NULL。

在一个线程中调用pthread_create()创建新的线程后,当前线程从pthread_create()返回继续往下执行,而新的线程所执行的代码由我们传给pthread_create的函数指针start_routine决定(线程创建成功后就立即去执行start_routine函数)。start_routine函数接收一个参数,是通过pthread_create的arg参数传递给它的,该参数的类型为void *,这个指针按什么类型解释由调用者自己定义。start_routine的返回值类型也是void *,这个指针的含义同样由调用者自己定义。start_routine返回时,这个线程就退出了,其它线程可以调用pthread_join得到start_routine的返回值,类似于父进程调用wait( )得到子进程的退出状态,稍后详细介绍pthread_join。

pthread_create成功返回后,新创建的线程的id被填写到thread参数所指向的内存单元。我们知道进程id的类型是pid_t,每个进程的id在整个系统中是唯一的,调用getpid( )可以获得当前进程的id,是一个正整数值。线程id的类型是pthread_t,它只在当前进程中保证是唯一的,在不同的系统中pthread_t这个类型有不同的实现,它可能是一个整数值,也可能是一个结构体,也可能是一个地址,所以不能简单地当成整数(%lu)用printf打印,调用pthread_self( )可以获得当前线程的id

attr参数表示线程属性,本节不深入讨论线程属性,所有代码例子都传NULL给attr参数,表示线程属性取缺省值,感兴趣的读者可以参考APUE。

//pthread_selfpthread_create函数示例

#include <stdio.h>
#include <pthread.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>void *tfn(void *arg)
{printf("In thread, the process id is %d and thread id is %lu.\n",getpid( ),pthread_self( ));return NULL;
}int main(void)
{pthread_t tid;int ret;printf("In main1, the process id is %d and thread id is %lu.\n",getpid( ),pthread_self( ));ret = pthread_create(&tid, NULL, tfn, NULL);if( ret != 0 )   //出错判断{fprintf(stderr,"pthread_create error: %s\n",strerror(ret));exit(1);}printf("the created thread's id is %lu\n",tid);sleep(1);  //注意一定要睡1sprintf("In main2, the process id is %u and thread id is %lu.\n",getpid( ),pthread_self( ));return 0;
}

[root@localhost 01_pthread_test]# ./pthrd_crt

In main1, the process id is 9814 and thread id is 4151932608.

the created thread's id is 4149869376

In thread, the process id is 9814 and thread id is 4149869376.

In main2, the process id is 9814 and thread id is 4151932608.

分析:

  1. main函数中调用sleep函数的原因。在Linux环境中,进程的本质就是含有一个线程的进程,其在创建线程之前就包含一个线程,因此此时有一个进程ID和一个线程ID(当然也有一个线程号),两个ID不想等,将此时的线程称为主控线程,后面创建的线程称为子线程。主线程创建完子线程后,如果主线程提前就结束了(执行了return或者exit等语句),则此时进程的地址空间会被回收,因此子线程也就被终止了(因为线程共享地址空间)。因此,主线程(main函数)必须要睡1s等待子线程完成工作后再结束。
  2. 注意线程库函数调用失败返回值的判断。出错不会再置ennro的值为相应的值,而是直接输出错误编号,因此不能再用perror函数来输出详细错误信息(该函数是根据errno的值输出相应的错误信息)。可以采用strerror函数,其作用:返回一个字符串,这个字符串描述了错误编号所对应的错误信息。char * strerror(int errnum);     如上所示:fprintf(stderr,"pthread_create error: %s\n",strerror(ret)); 采用fprintf函数将错误信息输出。相对于printf,fprintf函数可以将内容输出到指定文件中,第一个参数为文件的结构体指针。printf与fprintf都是标准输出,stdin、stdout和stderr为标准输入、标准输出和标准错误输出对应的文件结构体。
  3. 输出结果可以看出:在创建子线程前,进程本身包含一个线程;同一个进程中的所有线程的进程ID都一样;线程ID明显比线程号和进程ID大得多。

总结:由于pthread_create的错误码不保存在errno中,因此不能直接用perror( )打印错误信息,可以先用strerror( )把错误码转换成错误信息再打印。为了防止新创建的线程还没有得到执行就终止,我们在main函数return之前延时1秒,这只是一种权宜之计,即使主线程等待1秒,内核也不一定会调度新创建的线程执行,后面会有更好的办法。

//循环创建n个子线程的架构

#include <stdio.h>
#include <pthread.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>void *tfn(void *arg)
{int s = (int)arg;sleep(s);printf("The %dth thread: the process id is %d and thread id is %lu.\n",s+1,getpid( ),pthread_self( ));return NULL;
}int main(void)
{pthread_t tid;int ret, i;for( i=0;i<5;i++ ){ret = pthread_create(&tid, NULL, tfn,(void *)i);if( ret != 0 ){fprintf(stderr,"pthread_create error: %s\n",strerror(ret));exit(1);}}sleep(i);printf("In main: the process id is %u and thread id is %lu.\n",getpid( ),pthread_self( ));return 0;
}

[root@localhost 01_pthread_test]# ./pthrd_crt

The 1th thread: the process id is 4026 and thread id is 4149246784.

The 2th thread: the process id is 4026 and thread id is 4140854080.

The 3th thread: the process id is 4026 and thread id is 4132461376.

The 4th thread: the process id is 4026 and thread id is 4124068672.

The 5th thread: the process id is 4026 and thread id is 4115675968.

In main: the process id is 4026 and thread id is 4151466240.

分析:

  1. 如上循环创建5个线程,该进程中有6个线程,进程ID均相同,线程ID不一样;
  2. 每一个进程中线程的数目是有限的,不能超过这个值;
  3. 特别强调:主控线程调用的pthread_create函数中的第4个参数(void *)i不能改为:(void *)&i,相应在子线程执行的函数tfn中为s = *( (int *)arg ); 详细分析如下:首先,main函数是主控线程执行的函数,因此位于主控线程的用户栈帧空间中,该空间保存了main函数中的局部变量和形参值:tid、red、i,当main函数又去调用pthread_create函数时,main函数的栈帧头尾作为临时值保存在栈帧空间中。pthread_create函数的形参值保存在它的栈帧空间中。对于tfn函数(子线程执行的函数),它的局部变量和形参值(s和arg)又保存在各个线程的用户栈空间中,相互独立。外界参数的值传递给函数的形参时,都是按值传递(指针则传地址值,非指针变量则传变量值),如果传参数(void *)i,则将数值i转变为指针值 (指针值与数值i在大小上是相等的,只是在32位系统中指针值和整型i都占据4个字节;在64位系统中,指针值占据8个字节,而整型i依然占据4个字节,此时高位补0即可。然后将该指针值赋值给指针变量arg,即arg的值大小上是等于i的,但arg位于子线程的用户栈空间中,然后:(int)arg;即把指针值转变为整型s,s当然等于i,因此该方式是正确的。在64位系统中,指针值转变为整型i,8字节变为4字节,截取高位,而高位都是0,因此无影响。但如果是:(void *)&i  *( (int *)arg),这种情况下传递的是i的地址给arg,因此在子线程中是通过i的地址来访问数值i的,而数值i在主控线程中是会随时发生变化的(for循环中的i++导致),因此该方式不成立。
  4. pthread_create函数是一个回调函数,若创建子线程成功,则会去调用相应的函数;
  5. 子线程执行完自己的函数后返回的值通过pthread_join函数回收;
  6. return的作用是返回到函数的调用点,如果是main函数中的return,则代表该进程结束,并释放进程地址空间,所有线程都终止。对于其它函数的return,则直接返回到函数的调用点。exit和_exit函数会直接终止整个进程,导致所有线程结束。pthread_exit函数则会导致调用该函数的线程结束。所以,多线程环境中,应尽量少用,或者不使用exit函数,取而代之使用pthread_exit函数,将单个线程退出。任何线程里exit导致进程退出,其他线程也结束,主控线程退出时不能return或exit。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/385302.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

1005. 继续(3n+1)猜想 (25)

卡拉兹(Callatz)猜想已经在1001中给出了描述。在这个题目里&#xff0c;情况稍微有些复杂。 当我们验证卡拉兹猜想的时候&#xff0c;为了避免重复计算&#xff0c;可以记录下递推过程中遇到的每一个数。例如对n3进行验证的时候&#xff0c;我们需要计算3、5、8、4、2、1&#…

C指针深度解析

&#xff08;1&#xff09;指针的概念 指针是一种数据类型&#xff0c;而内存地址是这种数据类型具体的值&#xff08;注意区分两者的概念&#xff09;。先说一下什么是内存地址&#xff1a;假设CPU的寻址方式是以字节寻址的&#xff0c;即每一个字节对应一个地址编号&#xf…

1007. 素数对猜想

让我们定义 dn 为&#xff1a;dn pn1 - pn&#xff0c;其中 pi 是第i个素数。显然有 d11 且对于n>1有 dn 是偶数。“素数对猜想”认为“存在无穷多对相邻且差为2的素数”。 现给定任意正整数N (< 105)&#xff0c;请计算不超过N的满足猜想的素数对的个数。 输入格式&…

线程共享全局变量(.data和.bbs)

线程默认共享数据段、代码段等地址空间&#xff0c;常用的是全局变量。而进程不共享全局变量&#xff0c;只能借助mmap。 //代码示例 #include <string.h> #include <stdio.h> #include <pthread.h> #include <stdlib.h> #include <string.h> …

1008 数组元素循环右移问题 (20)

一个数组A中存有N&#xff08;N&gt0&#xff09;个整数&#xff0c;在不允许使用另外数组的前提下&#xff0c;将每个整数循环向右移M&#xff08;M>0&#xff09;个位置&#xff0c;即将A中的数据由&#xff08;A~0~ A~1~……A~N-1~&#xff09;变换为&#xff08;A~N-…

C++设计模式之策略模式(Strategy)

Strategy策略模式作用&#xff1a;定义了算法家族&#xff0c;分别封装起来&#xff0c;让他们之间可以互相替换&#xff0c;此模式让算法的变化&#xff0c;不会影响到使用算法的客户。 UML图&#xff1a; 代码实现 #include <iostream> using namespace std;class St…

pthread_exit函数

void pthread_exit(void *retval); 参数&#xff1a;retval表示线程退出状态&#xff0c;通常传NULL。 作用&#xff1a;将单个线程退出。 注意几点&#xff1a; return的作用是返回到函数的调用点&#xff0c;如果是main函数中的return&#xff0c;则代表该进程结束&#x…

C++面试常见问题

背景色yellow 1 1. c如何防止一个类被其他类继承 >- 如果是仅仅为了达到这个目的可以直接把这个类的构造函数设置成私有的&#xff0c;这样就杜绝了其他类的继承。也相当于毁掉了这个类&#xff08;无法再创造出自己的对象&#xff09;。 那么怎么样既要保证这个类的完整性…

pthread_join函数

int pthread_join(pthread_t thread, void **retval); 作用&#xff1a;阻塞等待线程退出&#xff0c;获取线程退出状态。其作用对应进程中 waitpid() 函数。 成功&#xff1a;0&#xff1b;失败&#xff1a;错误号 strerror函数 参数&#xff1a;thread&#xff1a;线程I…

【C++ Priemr | 15】派生类向基类转换的可访问性

1. 只有当D公有继承B时&#xff0c;用户代码才能使用派生类向基类的转换&#xff1b;如果D私有继承B的方式是受保护的或者私有的&#xff0c;则用户代码不能使用该转换。 class A {}&#xff1b; class B : public A {}void function(const A&) {}int main() {B b;functio…

pthread_detach函数

int pthread_detach(pthread_t thread); 成功&#xff1a;0&#xff1b;失败&#xff1a;错误号 作用&#xff1a;从状态上实现线程分离&#xff0c;注意不是指该线程独自占用地址空间。 线程分离状态&#xff1a;指定该状态&#xff0c;线程主动与主控线程断开关系。线程…

【C++ Primer | 15】面试问题

在成员函数中调用虚函数 #include <iostream> using namespace std; class CBase { public:void func1(){func2();}virtual void func2() {cout << "CBase::func2()" << endl;} }; class CDerived:public CBase { public:virtual void func2() {…

pthread_cancel、pthread_equal函数

&#xff08;1&#xff09;pthread_cancel函数 int pthread_cancel(pthread_t thread); 成功&#xff1a;0&#xff1b;失败&#xff1a;错误号 作用&#xff1a;杀死(取消)线程&#xff0c;其作用对应进程中 kill() 函数。 注意&#xff1a;线程的取消并不是实时的&…

STL源码剖析面试问题

当vector的内存用完了&#xff0c;它是如何动态扩展内存的&#xff1f;它是怎么释放内存的&#xff1f;用clear可以释放掉内存吗&#xff1f;是不是线程安全的&#xff1f; vector内存用完了&#xff0c;会以当前size大小重新申请2* size的内存&#xff0c;然后把原来的元素复制…

线程与进程的控制原语对比

线程与进程的控制原语对比 fork pthead_create exit( int ) pthead_exit(void *); wait(int *) pthread_join&#xff08; ,void **&#xff09; 阻塞 ;分离 22 &#xff1b;cancel -1 kill() pthread_cancel(); 取消点(检查点)&#xff1a;系统调用 getpid() pthrea…

ptmalloc堆内存管理机制(主要讨论Linux x86下32位系统)

bin&#xff08;chunk容器&#xff09; ptmalloc将相似大小的 chunk 用双向链表链接起来&#xff0c;这样的一个链表被称为一个 bin。 Ptmalloc 一共维护了 128 个 bin&#xff0c;并使用一个数组来存储这些 bin&#xff0c;这个数组被成为bin数组。 bin数组结构如下&#xf…

线程属性的修改

&#xff08;1&#xff09;线程属性 Linux下线程的属性是可以根据实际项目需要&#xff0c;进行设置&#xff0c;之前我们讨论的线程都是采用线程的默认属性&#xff0c;默认属性已经可以解决绝大多数开发时遇到的问题。如我们对程序的性能提出更高的要求那么需要设置线程属性…

NPTL(Native POSIX Thread Library)

1.NPTL&#xff08;Native POSIX Thread Library&#xff09;为POSIX标准线程库&#xff0c;查看当前Linux系统的pthread库&#xff08;线程库&#xff09;版本的命令为&#xff1a;getconf GNU_LIBPTHREAD_VERSION。 [rootlocalhost 01_pthread_test]# getconf GNU_LIBPTHREA…

线程使用注意事项

1.主线程退出其他线程不退出&#xff0c;主线程应调用pthread_exit&#xff1b; 2.避免僵尸线程&#xff1a;pthread_join、pthread_detach、pthread_create指定分离属性。被join线程可能在join函数返回前就释放完自己的所有内存资源&#xff0c;所以不应当返回被回收线程栈中…

线程同步的概念

所谓同步&#xff0c;即同时起步&#xff0c;协调一致。不同的对象&#xff0c;对“同步”的理解方式略有不同。如&#xff0c;设备同步&#xff0c;是指在两个设备之间规定一个共同的时间参考&#xff1b;数据库同步&#xff0c;是指让两个或多个数据库内容保持一致&#xff0…