进程号在系统中唯一,但线程号只在其所属进程环境中有效。
1 2 3 4 5 6 7 8 9 |
#include<pthread.h>
pthread_t pthread_self(void); /* 功能: 获取线程号 返回值: 调用此函数线程的ID */ |
pthread_self示例:
1 2 3 4 5 6 7 8 9 |
#include<stdio.h> #include<pthread.h>
int main(int argc, const char* argv[]) { pthread_t tid = 0; tid = pthread_self(); printf("当前线程id:%lu.\n", tid); return 0; } |
编译时需要加上-pthread链接到pthread库
运行结果:
1 2 3 4 5 6 7 8 |
int pthraed_equal(pthread_t t1, pthread_t t2); /* 功能: 判断线程号t1、t2是否相等。 返回值: 相等:非0 不等:0 */ |
pthread_equal示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
#include<stdio.h> #include<pthread.h>
int main(int argc, const char* argv[]) { pthread_t tid = 0; tid = pthread_self();
if (pthread_equal(tid, pthread_self())) { printf("线程id相等.\n"); } else { printf("线程id不等.\n"); } return 0; } |
运行结果:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
#include<pthread.h>
int pthread_create(pthread_t* thread, const pthread_attr_t* attr, void* (*start_coutine)(void*), void* arg); /* 功能: 创建一个线程。 参数: thread:线程id地址,为传出参数; attr:线程属性结构体,通常设置为NULL; start_routine:线程函数入口地址 arg:传给线程函数的参数; 返回值: 成功:0 失败:非0,未设置errno,不可使用perror。 */ |
pthread_create示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 |
#include<stdio.h> #include<pthread.h> #include<unistd.h>
void* threadFunc(void* arg) { // 线程调度函数 int var = (int)(long)(arg); printf("被创建线程id:%lu,传过来的参数:%d\n", pthread_self(), var); return NULL; }
int main(int argc, const char* argv[]) {
pthread_t tid; int ret = -1;
// 初始化tid。因为不是所有系统中的pthread_t都是unsigned int, 因此最好使用memset初始化。 memset(&tid, 0, sizeof(tid));
// 创建线程 ret = pthread_create(&tid, NULL, threadFunc, (void*)0x3); if (0 != ret) { printf("线程创建失败!\n"); return 1; } printf("按下任意键继续...\n"); getchar(); printf("主线程id:%lu\n", pthread_self());
return 0; } |
运行结果:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
typedef struct { int etachstate; // 线程的分离状态 int schedpolicy; // 线程的调度策略 struct sched_param schedparam; // 线程的调度参数 int inheritsched; // 线程的继承性 int scope; // 线程的作用域 size_t guardsize; // 线程栈末尾的警戒缓冲区大小 int stackaddr_set; // 线程栈的设置 void* stackaddr; // 线程栈的位置 size_t stacksize; // 线程栈的大小 } pthread_attr_t; /* 功能: 线程属性结构体; 主要成员: etachstate:线程的分离状态 guardsize:线程栈末尾的警戒缓冲区大小 stackaddr:线程栈的位置 stacksize:线程栈的大小
注意: 线程属性值不能直接设置,需使用相关函数进行操作。 如pthread_create之前用pthread_attr_init初始化, 之后用pthread_attr_destory释放资源。 */ |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
#include<pthread.h>
int pthread_attr_init(pthread_attr_t* attr); /* 功能: 初始化线程属性attr。 参数: attr:待初始化的线程属性结构体。 返回值: 成功:0 失败:错误码 */
int pthread_attr_destory(pthread_attr_t* attr);/* 功能: 销毁线程属性attr。 参数: attr:线程属性结构体。 返回值: 成功:0 失败:非0错误码 */ |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
#include<pthread.h>
int pthread_attr_setdetachstate(pthread_attr_t* attr, int detachstate); /* 功能: 设置线程属性为分离。 参数: attr:已初始化的线程属性结构体; detachstate:是否分离: PTHREAD_CREATE_DETACHED:分离 PTHREAD_CREATE_JOINABLE:非分离 返回值: 成功:0 失败:非0 */
int pthread_attr_getdetachstate(const pthread_attr_t* attr, int* detachstate); /* 功能: 获取线程是否分离状态 参数: attr:线程属性结构体。 detachstate:是否分离: PTHREAD_CREATE_DETACHED:分离 PTHREAD_CREATE_JOINABLE:非分离 返回值: 成功:0 失败:非0错误码 */ |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 |
#include<stdio.h> #include<stdlib.h> #include<string.h> #include<unistd.h> #include<pthread.h>
void* func(void* arg) { printf("子线程开始...\n"); sleep(2); printf("子线程结束...\n"); pthread_exit(NULL); }
int main(int argc, const char* argv[]) {
int ret = -1; pthread_t tid = -1; pthread_attr_t attr;
// 初始化线程属性 ret = pthread_attr_init(&attr); if (0 != ret) { printf("线程属性初始化失败。\n"); return 1; }
// 设置线程属性为分离状态 ret = pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); if (0 != ret) { printf("线程分离属性设置失败。\n"); return 1; }
// 创建线程 ret = pthread_create(&tid, &attr, func, NULL); if (0 != ret) { printf("线程创建失败.\n"); return 1; }
sleep(3);
// join看看是否为分离状态 ret = pthread_join(tid, NULL); if (0 != ret) { printf("线程为分离状态,无需join.\n"); } else { printf("线程为非分离状态,已被join.\n"); }
// 销毁线程属性 ret = pthread_attr_destroy(&attr); if (0 != ret) { printf("线程属性销毁失败。\n"); return 1; }
return 0; } |
运行结果:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
#include<pthread.h>
int pthread_attr_setstacksize(pthread_attr_t* attr, size_t stacksize); /* 功能: 设置线程栈大小。 参数: attr:线程属性结构体; stacksize:线程栈大小; 返回值: 成功:0; 失败:错误码 */
int pthread_attr_getstacksize(const pthread_attr_t* attr, size_t* stacksize); /* 功能: 获取线程栈大小。 参数: attr:线程属性结构体指针; stacksize:返回的线程栈大小; 返回值: 成功:0; 失败:非0错误码 */ |
主线程回收线程资源,会阻塞。
1 2 3 4 5 6 7 8 9 10 11 12 13 |
#include<pthread.h>
int pthread_join(pthread_t thread, void** retval); /* 功能: 类似于wait()函数。等待线程thread结束,回收线程资源;若线程已结束,则会立即返回。 参数: thread:等待回收的线程号; retval:存储进程退出状态的指针的地址; 返回值: 成功:0 失败:非0错误码 */ |
pthread_join示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 |
#include<stdio.h> #include<stdlib.h> #include<string.h> #include<pthread.h> #include<unistd.h>
void* func() { printf("子线程开始执行...\n"); sleep(3); printf("子线程结束执行...\n"); return (void*)0x3; }
int main(int argc, const char* argv[]) {
pthread_t tid; int ret = -1; void* retp = NULL;
memset(&tid, 0, sizeof(tid));
// 创建线程 ret = pthread_create(&tid, NULL, func, NULL); if (0 != ret) { printf("线程创建失败.\n"); return 1; }
printf("主线程执行...\n");
// 等待线程结束 pthread_join会阻塞 ret = pthread_join(tid, &retp); if (0 != ret) { printf("线程join失败.\n"); return 1; }
printf("retp: %p\n", retp); printf("主线程退出...\n");
return 0; } |
运行结果:
内核回收线程资源,不会阻塞。
1 2 3 4 5 6 7 8 9 10 11 12 |
#include<pthread.h>
int pthread_detach(pthread_t thread); /* 功能: 使线程thread与当前进程分离,之后线程结束后的资源回收由内核完成,因此不会阻塞。 参数: thread:线程号; 返回值: 成功:0 失败:非0 */ |
pthread_detach示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 |
#include<stdio.h> #include<stdlib.h> #include<string.h> #include<unistd.h> #include<pthread.h>
void* func(void* arg) { printf("子线程开始...\n"); for (int i = 0; i < 3;++i) { sleep(1); printf("子线程工作%ds\n", i); } printf("子线程结束...\n"); return NULL; }
int main(int argc, const char* argv[]) {
int ret = -1; pthread_t tid = -1;
// 创建线程 ret = pthread_create(&tid, NULL, func, NULL); if (0 != ret) { printf("线程创建失败.\n"); return 1; }
// 设置线程分离 ret = pthread_detach(tid); if (0 != ret) { printf("线程分离失败.\n"); return 1; }
printf("主线程:按回车键退出..\n"); getchar();
return 0; } |
运行结果:
线程设置为detach状态,主线程不必阻塞等待回收子线程资源,而是由内核完成。
若在线程中用exit函数退出,则导致整个进程退出,而非退出这一个线程。
如下三者可在不结束整个进程的情况下结束线程:
a)线程从执行函数中返回;
b)线程调用pthread_exit退出线程;
c)线程被同一进程中的其它线程取消。
1 2 3 4 5 6 7 8 9 |
#include<pthread.h>
void pthread_exit(void* retval); /* 功能: 退出调用线程。 参数: retval:存储线程退出状态的指针。 */ |
1 2 3 4 5 6 7 8 9 10 11 12 |
#include<pthread.h>
int pthread_cancel(pthread_t thread); /* 功能: 杀死线程thread; 参数: thread:目标线程ID; 返回值: 成功:0 失败:非0错误码。 */ |
pthread_cancel示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 |
#include<stdio.h> #include<stdlib.h> #include<string.h> #include<unistd.h> #include<pthread.h>
void* func(void* arg) { printf("子线程开始...\n"); for (int i = 0; i < 5;++i) { sleep(1); printf("子线程工作%ds\n", i); } printf("子线程结束...\n"); pthread_exit(NULL); }
int main(int argc, const char* argv[]) {
int ret = -1; pthread_t tid = -1;
// 创建线程 ret = pthread_create(&tid, NULL, func, NULL); if (0 != ret) { printf("线程创建失败.\n"); return 1; }
// 设置线程分离 ret = pthread_detach(tid); if (0 != ret) { printf("线程分离失败.\n"); return 1; }
sleep(3); pthread_cancel(tid); // 杀死线程
printf("主线程:按回车键退出..\n"); getchar();
return 0; } |
运行结果:
(1)主线程退出,而其余线程不退出,主线程应调用pthread_exit;
(2)避免僵尸线程方式:
(3)malloc和mmap申请的内存可被其他线程释放;
(4)避免在多线程中fork,除非马上使用exec;
(5)避免多线程中使用信号机制。