多线程实现多任务

概述

每一个进程都拥有自己的数据段、代码段和堆栈段,这就造成进程在进行创建、切换、撤销操作时,须要较大的系统开销。

为了降低系统开销,从进程中演化出了线程。

为了让进程完毕一定的工作,进程必须至少包括一个线程。线程存在于进程中,共享进程的资源。很多其它详情。请看《进程和线程的差别与联系》。

就像每一个进程都有一个进程号一样,每一个线程也有一个线程号。进程号在整个系统中是唯一的,但线程号不同,线程号仅仅在它所属的进程环境中有效。进程号用 pid_t 数据类型表示,是一个非负整数。线程号则用 pthread_t 数据类型来表示,Linux 使用无符号长整数表示。有的系统在实现 pthread_t 的时候,用一个结构体来表示,所以在可移植的操作系统实现不能把它做为整数处理。

线程的经常使用函数

1)获取线程号

所需头文件:

#include <pthread.h>

pthread_t pthread_self(void);

功能:

获取线程号。

參数:

返回值:

调用线程的线程 ID 。

2)线程号的比較

所需头文件:

#include <pthread.h>

int pthread_equal(pthread_t t1, pthread_t t2);

功能:

推断线程号 t1 和 t2 是否相等。

为了方便移植,尽量使用函数来比較线程 ID。

參数:

t1,t2:待推断的线程号。

返回值:

相等:  非 0

不相等:0

演示样例代码:

[cpp] view
plain
copy

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. #include <pthread.h>
  4. int main(int argc, char *argv[])
  5. {
  6. pthread_t thread_id;
  7. thread_id = pthread_self(); // 返回调用线程的线程ID
  8. printf("Thread ID = %lu \n",thread_id);
  9. if( 0 != pthread_equal( thread_id, pthread_self() ) ){
  10. printf("Equal!\n");
  11. }else{
  12. printf("Not equal!\n");
  13. }
  14. return 0;
  15. }

线程函数的程序在 pthread 库中,故链接时要加上參数 -lpthread

执行结果例如以下:

3)线程的创建

所需头文件:

#include <pthread.h>

int pthread_create( pthread_t *thread,

const pthread_attr_t *attr,

void *(*start_routine)(void *),

void *arg );

功能:

创建一个线程。

參数:

thread:线程标识符地址。

attr:线程属性结构体地址,通常设置为 NULL。

start_routine:线程函数的入口地址。

arg:传给线程函数的參数。

返回值:

成功:0

失败:非 0

pthread_create() 创建的线程从指定的回调函数開始执行,该函数执行完后,该线程也就退出了。

线程依赖进程存在的,共享进程的资源,假设创建线程的进程结束了。线程也就结束了。

演示样例一:

[cpp] view
plain
copy

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <pthread.h>
  4. int var  = 8;
  5. void *thread_1(void *arg)
  6. {
  7. while(1)
  8. {
  9. printf("this is my new thread1: var++\n");
  10. var++;
  11. sleep(1);
  12. }
  13. return NULL;
  14. }
  15. void *thread_2(void * arg)
  16. {
  17. while(1){
  18. printf("this is my new thread2: var = %d\n", var);
  19. sleep(1);
  20. }
  21. return NULL;
  22. }
  23. int main(int argc, char *argv[])
  24. {
  25. pthread_t tid1,tid2;
  26. //创建两个线程
  27. pthread_create(&tid1, NULL, thread_1, NULL);
  28. pthread_create(&tid2, NULL, thread_2, NULL);
  29. while(1){
  30. printf("the main thread: var = %d\n", var);
  31. sleep(1);
  32. }
  33. return 0;
  34. }

执行结果例如以下:

演示样例二:

[cpp] view
plain
copy

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <pthread.h>
  4. // 回调函数
  5. void *thread_fun(void * arg)
  6. {
  7. sleep(1);
  8. int num = *( (int *)arg );
  9. printf("int the new thread: num = %d\n", num);
  10. return NULL;
  11. }
  12. int main(int argc, char *argv[])
  13. {
  14. pthread_t tid;
  15. int test = 100;
  16. // 创建线程, 把 &test 传给回调函数 thread_fun()
  17. pthread_create(&tid, NULL, thread_fun, (void *)&test);
  18. while(1);
  19. return 0;
  20. }

执行结果例如以下:

4)回收线程资源

所需头文件:

#include <pthread.h>

int pthread_join(pthread_t thread, void **retval);

功能:

等待线程结束(此函数会堵塞),并回收线程资源,类似进程的 wait() 函数。假设线程已经结束。那么该函数会马上返回。

參数:

thread:被等待的线程号。

retval:用来存储线程退出状态的指针的地址。

返回值:

成功:0

失败:非 0

演示样例代码例如以下:

[cpp] view
plain
copy

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <pthread.h>
  4. void *thead(void *arg)
  5. {
  6. static int num = 123; //静态变量
  7. printf("after 2 seceonds, thread will return\n");
  8. sleep(2);
  9. return #
  10. }
  11. int main(int argc, char *argv[])
  12. {
  13. pthread_t tid;
  14. int ret = 0;
  15. void *value = NULL;
  16. // 创建线程
  17. ret = pthread_create(&tid, NULL, thead, NULL);
  18. if(ret != 0){ //创建失败
  19. perror("pthread_create");
  20. }
  21. // 等待线程号为 tid 的线程,假设此线程结束就回收其资源
  22. // &value保存线程退出的返回值
  23. pthread_join(tid, &value);
  24. printf("value = %d\n", *( (int *)value ) );
  25. return 0;
  26. }

执行结果例如以下:

创建一个线程后应回收其资源,但使用 pthread_join() 函数会使调用者堵塞,Linux 还提供了非堵塞函数 pthread_detach() 来回收线程的资源。

所需头文件:

#include <pthread.h>

int pthread_detach(pthread_t thread);

功能:

使调用线程与当前进程分离,分离后不代表此线程不依赖与当前进程,线程分离的目的是将线程资源的回收工作交由系统自己主动来完毕,也就是说当被分离的线程结束之后,系统会自己主动回收它的资源。所以。此函数不会堵塞。

參数:

thread:线程号。

返回值:

成功:0

失败:非 0

注意,调用 pthread_detach() 后再调用 pthread_join() , pthread_join() 会立刻返回,调用失败。

演示样例代码例如以下:

[cpp] view
plain
copy

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <pthread.h>
  4. void *thead(void *arg)
  5. {
  6. int i;
  7. for(i=0; i<5; i++)
  8. {
  9. printf("I am runing\n");
  10. sleep(1);
  11. }
  12. return NULL;
  13. }
  14. int main(int argc, char *argv[])
  15. {
  16. int ret  = 0;
  17. pthread_t tid;
  18. ret = pthread_create(&tid, NULL, thead, NULL);
  19. if(ret!=0){
  20. perror("pthread_create");
  21. }
  22. pthread_detach(tid); // 线程分离。不堵塞
  23. // 立刻返回,调用失败
  24. int flag = pthread_join(tid, NULL);
  25. if(flag != 0){
  26. printf("join not working\n");
  27. }
  28. printf("after join\n");
  29. sleep(3);
  30. printf("I am leaving\n");
  31. return 0;
  32. }

执行结果例如以下:

5)线程退出

在进程中我们能够调用 exit() 函数或 _exit() 函数来结束进程。在一个线程中我们能够通过 pthread_exit() 在不终止整个进程的情况下停止它的控制流。

所需头文件:

#include <pthread.h>

void pthread_exit(void *retval);

功能:

退出调用线程。一个进程中的多个线程是共享该进程的数据段,因此。通常线程退出后所占用的资源并不会释放。

參数:

retval:存储线程退出状态的指针。

返回值:

演示样例代码例如以下:

[cpp] view
plain
copy

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <pthread.h>
  4. void *thread(void *arg)
  5. {
  6. static int num = 123; //静态变量
  7. int i = 0;
  8. while(1)
  9. {
  10. printf("I am runing\n");
  11. sleep(1);
  12. i++;
  13. if(i==3)
  14. {
  15. pthread_exit( (void *)&num );
  16. // return &num;
  17. }
  18. }
  19. return NULL;
  20. }
  21. int main(int argc, char *argv[])
  22. {
  23. int ret  = 0;
  24. pthread_t tid;
  25. void *value  = NULL;
  26. ret = pthread_create(&tid, NULL, thread, NULL);
  27. if(ret!=0){
  28. perror("pthread_create");
  29. }
  30. pthread_join(tid, &value);
  31. printf("value = %d\n", *(int *)value );
  32. return 0;
  33. }

执行结果例如以下:

本教程演示样例代码下载请点此链接。

转自:

时间: 2024-10-22 09:09:47

多线程实现多任务的相关文章

python多线程实现多任务

1.什么是线程? 进程是操作系统分配程序执行资源的单位,而线程是进程的一个实体,是CPU调度和分配的单位.一个进程肯定有一个主线程,我们可以在一个进程里创建多个线程来实现多任务. ----------------------------------------------------------------------------------------------------------------------------------------------------------------

Linux系统编程——多线程实现多任务

概述 每个进程都拥有自己的数据段.代码段和堆栈段,这就造成进程在进行创建.切换.撤销操作时,需要较大的系统开销.为了减少系统开销,从进程中演化出了线程.为了让进程完成一定的工作,进程必须至少包含一个线程.线程存在于进程中,共享进程的资源.更多详情,请看<进程和线程的区别与联系>. 就像每个进程都有一个进程号一样,每个线程也有一个线程号.进程号在整个系统中是唯一的,但线程号不同,线程号只在它所属的进程环境中有效.进程号用 pid_t 数据类型表示,是一个非负整数.线程号则用 pthread_t

.NET多线程编程(1)——多任务和多线程

在.NET多线程编程这个系列我们讲一起来探讨多线程编程的各个方面.首先我将在本篇文章的开始向大家介绍多线程的有关概念以及多线程编程的基础知识;在接下来的文章中,我将逐一讲述.NET平台上多线程编程的知识,诸如System.Threading命名空间的重要类以及方法,并就一些例子程序来作说明. 引言 早期的计算硬件十分复杂,但是操作系统执行的功能确十分的简单.那个时候的操作系统在任一时间点只能执行一个任务,也就是同一时间只能执行一个程序.多个任务的执行必须得轮流执行,在系统里面进行排队等候.由于计

09 多线程

作者:thoughtInShare 出处:http://www.cnblogs.com/thoughtInShare 欢迎转载,也请保留这段声明.谢谢! 1,进程是处于运行中的程序,并具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位: 线程是轻量级的进程,线程是进程的执行单元: 一个进程可以有多个线程,一个线程必有一个父进程: 线程可以拥有自己的栈,以及程序计数器,但不拥有系统资源:它与父进程的其他线程共享进程拥有的全部资源: 线程是独立的,它并不知道进程中是否有其他线程存在: 线

Java 并发和多线程(一) Java并发性和多线程介绍[转]

作者:Jakob Jenkov 译者:Simon-SZ  校对:方腾飞 http://tutorials.jenkov.com/java-concurrency/index.html 在过去单CPU时代,单任务在一个时间点只能执行单一程序.之后发展到多任务阶段,计算机能在同一时间点并行执行多任务或多进程.虽然并不是真正意义上的“同一时间点”,而是多个任务或进程共享一个CPU,并交由操作系统来完成多任务间对CPU的运行切换,以使得每个任务都有机会获得一定的时间片运行. 随着多任务对软件开发者带来的

.NET多线程编程

在.NET多线程编程这个系列我们讲一起来探讨多线程编程的各个方面.首先我将在本篇文章的开始向大家介绍多线程的有关概念以及多线程编程的基础知识;在接下来的文章中,我将逐一讲述.NET平台上多线程编程的知识,诸如System.Threading命名空间的重要类以及方法,并就一些例子程序来作说明. 引言 早期的计算硬件十分复杂,但是操作系统执行的功能确十分的简单.那个时候的操作系统在任一时间点只能执行一个任务,也就是同一时间只能执行一个程序.多个任务的执行必须得轮流执行,在系统里面进行排队等候.由于计

5天玩转C#并行和多线程编程 —— 第五天 多线程编程大总结

5天玩转C#并行和多线程编程系列文章目录 5天玩转C#并行和多线程编程 —— 第一天 认识Parallel 5天玩转C#并行和多线程编程 —— 第二天 并行集合和PLinq 5天玩转C#并行和多线程编程 —— 第三天 认识和使用Task 5天玩转C#并行和多线程编程 —— 第四天 Task进阶  一.多线程带来的问题 1.死锁问题  前面我们学习了Task的使用方法,其中Task的等待机制让我们瞬间爱上了它,但是如果我们在调用Task.WaitAll方法等待所有线程时,如果有一个Task一直不返

Java并发性和多线程介绍

以下内容转自http://ifeve.com/java-concurrency-thread/: 在过去单CPU时代,单任务在一个时间点只能执行单一程序.之后发展到多任务阶段,计算机能在同一时间点并行执行多任务或多进程.虽然并不是真正意义上的“同一时间点”,而是多个任务或进程共享一个CPU,并交由操作系统来完成多任务间对CPU的运行切换,以使得每个任务都有机会获得一定的时间片运行. 随着多任务对软件开发者带来的新挑战,程序不在能假设独占所有的CPU时间.所有的内存和其他计算机资源.一个好的程序榜

一步步学习Linux多任务编程(陆续更新)

系统调用 01.什么是系统调用? 02.Linux系统调用之I/O操作(文件操作) 03.文件描述符的复制:dup(), dup2() 多进程实现多任务 04.进程的介绍 05.Linux可执行文件结构与进程结构 06.多进程实现多任务(一):fork() 07.多进程实现多任务(二):vfork() 08.进程的控制:结束进程.等待进程结束 09.Linux特殊进程之僵尸进程 10.Linux特殊进程之孤儿进程 11.Linux特殊进程之守护进程 12.进程替换:exec函数族 进程间通信 1