多线程编程之:Linux线程编程
9.2 Linux线程编程
9.2.1 线程基本编程
这里要讲的线程相关操作都是用户空间中的线程的操作。在Linux中,一般pthread线程库是一套通用的线程库,是由POSIX提出的,因此具有很好的可移植性。
(1)函数说明。
创建线程实际上就是确定调用该线程函数的入口点,这里通常使用的函数是pthread_create()。在线程创建以后,就开始运行相关的线程函数,在该函数运行完之后,该线程也就退出了,这也是线程退出一种方法。另一种退出线程的方法是使用函数pthread_exit(),这是线程的主动行为。这里要注意的是,在使用线程函数时,不能随意使用exit()退出函数进行出错处理,由于exit()的作用是使调用进程终止,往往一个进程包含多个线程,因此,在使用exit()之后,该进程中的所有线程都终止了。因此,在线程中就可以使用pthread_exit()来代替进程中的exit()。
由于一个进程中的多个线程是共享数据段的,因此通常在线程退出之后,退出线程所占用的资源并不会随着线程的终止而得到释放。正如进程之间可以用wait()系统调用来同步终止并释放资源一样,线程之间也有类似机制,那就是pthread_join()函数。pthread_join()可以用于将当前线程挂起来等待线程的结束。这个函数是一个线程阻塞的函数,调用它的函数将一直等待到被等待的线程结束为止,当函数返回时,被等待线程的资源就被收回。
前面已提到线程调用pthread_exit()函数主动终止自身线程。但是在很多线程应用中,经常会遇到在别的线程中要终止另一个线程的执行的问题。此时调用pthread_cancel()函数实现这种功能,但在被取消的线程的内部需要调用pthread_setcancel()函数和pthread_setcanceltype()函数设置自己的取消状态,例如被取消的线程接收到另一个线程的取消请求之后,是接受还是忽略这个请求;如果接受,是立刻进行终止操作还是等待某个函数的调用等。
(2)函数格式。
表9.1列出了pthread_create()函数的语法要点。
表9.2列出了pthread_exit()函数的语法要点。
表9.3列出了pthread_join()函数的语法要点。
表9.4列出了pthread_cancel()函数的语法要点。
(3)函数使用。
以下实例中创建了3个线程,为了更好地描述线程之间的并行执行,让3个线程重用同一个执行函数。每个线程都有5次循环(可以看成5个小任务),每次循环之间会随机等待1~10s的时间,意义在于模拟每个任务的到达时间是随机的,并没有任何特定规律。
/* thread.c */
#include
#include
#include
#define THREAD_NUMBER 3 /*线程数*/
#define REPEAT_NUMBER 5 /*每个线程中的小任务数*/
#define DELAY_TIME_LEVELS 10.0 /*小任务之间的最大时间间隔*/
void *thrd_func(void *arg)
{ /* 线程函数例程 */
int thrd_num = (int)arg;
int delay_time = 0;
int count = 0;
printf("Thread %d is starting\n", thrd_num);
for (count = 0; count < REPEAT_NUMBER; count++)
{
delay_time = (int)(rand() * DELAY_TIME_LEVELS/(RAND_MAX)) + 1;
sleep(delay_time);
printf("\tThread %d: job %d delay = %d\n",
thrd_num, count, delay_time);
}
printf("Thread %d finished\n", thrd_num);
pthread_exit(NULL);
}
int main(void)
{
pthread_t thread[THREAD_NUMBER];
int no = 0, res;
void * thrd_ret;
srand(time(NULL));
for (no = 0; no < THREAD_NUMBER; no++)
{
/* 创建多线程*/
res = pthread_create(&thread[no], NULL, thrd_func, (void*)no);
if (res != 0)
{
printf("Create thread %d failed\n", no);
exit(res);
}
}
printf("Create treads success\n Waiting for threads to finish...\n");
for (no = 0; no < THREAD_NUMBER; no++)
{
/* 等待线程结束 */
res = pthread_join(thread[no], &thrd_ret);
if (!res)
{
printf("Thread %d joined\n", no);
}
else
{
printf("Thread %d join failed\n", no);
}
}
return 0;
}
以下是程序运行结果。可以看出每个线程的运行和结束是独立与并行的。
$ ./thread
Create treads success
Waiting for threads to finish...
Thread 0 is starting
Thread 1 is starting
Thread 2 is starting
Thread 1: job 0 delay = 6
Thread 2: job 0 delay = 6
Thread 0: job 0 delay = 9
Thread 1: job 1 delay = 6
Thread 2: job 1 delay = 8
Thread 0: job 1 delay = 8
Thread 2: job 2 delay = 3
Thread 0: job 2 delay = 3
Thread 2: job 3 delay = 3
Thread 2: job 4 delay = 1
Thread 2 finished
Thread 1: job 2 delay = 10
Thread 1: job 3 delay = 4
Thread 1: job 4 delay = 1
Thread 1 finished
Thread 0: job 3 delay = 9
Thread 0: job 4 delay = 2
Thread 0 finished
Thread 0 joined
Thread 1 joined
Thread 2 joined
您可能也感兴趣:
为您推荐
保险业去年原保费收入微降0.79% 健康险业务增长3.36%
银保监会:严防银行保险资金被用于盲目“加杠杆”,强化金融反垄断
保险打工人年终奖多数和上年持平或下降,怎样发才合理?
排行
最近更新
- 多线程编程之:Linux线程编程
- 加法器是什么?加法器电路原理
- 中国品牌TECNO Mobile手机在非洲有多火?
- 为什么电网中存在有功功率与无功功率?
- 电容式触摸屏原理
- 说唱歌手TYGA将Instagram头像更换为INNOCENT CATS NFT
- 比特币非零钱包数量突破4000万 创历史新高
- 空气治理概念股名单,空气治理概念股龙头有哪些?
- A股建材龙头上市公司汇总(附股)
- 新鲜出炉!碳中和概念龙头股全名单揭晓!(干货满满)
- 饲料龙头股上市公司有哪些,饲料龙头概念股票一览
- 人脑工程概念股有哪些,人脑工程概念股名单
- 朗源股份(300175)最大股东是谁?
- 电池容量的计算
- 常用稳压二极管技术参数
- 应变仪的种类有哪些?应变仪的常见类型
- 与非门组成振荡电路图
- 超强科普帖:萨德到底是个什么鬼?
- 触摸屏原理及基础知识
- 图文详解:LED驱动电路的基本架构和特性
- 华为拿出百亿给员工分红,任正非股权还不到1%
- 京东到家:平台异地订单同比增长1.3倍,预制菜较为走俏
- 新城控股计划提前赎回2亿美元债券 并予以注销
- 2022年A股市场有望延续结构性行情 港股反弹可期
- 人民网:数字人民币走进冬奥 非接触支付凸显防疫优势
- 以太坊Layer2总锁仓量突破60亿美元
- 百融云创推出智能语音技术 提速农村金融服务效率
- 可以影响全人类的雪莲养护技术诞生,大众日常生活用品增添新品类
- 中国创造雪莲养护技术走在世界前列,人体养护领域迎来国际话语权
- 生物质能概念股一览, 2021年生物质能龙头概念股有哪些