一、线程概念
进程
进程是程序的一次执行
线程
线程(thread)是操作系统能够进行运算调度的最小单位。
它被包含在进程之中,是进程中的实际运作单位。一个线程指的是进程中一个单一顺序的控制流,一个进程中可以并发多个线程,每条线程并行执行不同的任务。
线程是独立调度和分派的基本单位。
同一进程中的多个线程将共享该进程中的全部系统资源,如虚拟地址空间,文件描述符和信号处理等等。但同一进程中的多个线程有各自的调用栈(call stack),自己的寄存器环境(register context),自己的线程本地存储(thread-local storage)。
一个进程可以有很多线程,每个线程并行执行不同的任务。
二、Linux下创建线程
编写Linux下的多线程程序,需要使用头文件pthread.h
与线程相关的函数如下
函数名称 | 作用 | 备注 |
---|---|---|
int pthread_create(pthread_t *tidp, const pthread_attr_t *attr, func(void), arg) | 创建函数,若成功会返回0 | |
pthread_t | 线程ID数据类型,线程ID只在它所属的进程环境中有效 | 并不是函数 |
pthread_t pthread_self(void) | 获取自身线程ID | |
void pthread_exit(void *rval_ptr) | 终止本线程,用rval_ptr指向的值作为退出码 | |
int pthread_join(pthread_t thread, void ** rval_ptr) | 阻塞线程,成功返回0。以阻塞的方式等待thread指定的线程结束 | |
int pthread_cancel(pthread_t tid) | 请求取消统一进程中的其他线程 | 似乎不常用 |
函数详解
1.创建函数——pthread_create
该函数可创建一个新的线程,如果成功则返回0,否则返回错误代码
int pthread_create(pthread_t *tidp, const pthread_attr_t *attr, func(void), arg)
参数 | 性质 |
---|---|
1 pthread_t *tidp | 保存成功创建线程之后对应的线程id |
2 const pthread_attr_t *attr | 表示线程的属性(是一个结构体),通常用NULL会设置成为默认的属性,如果想使用具体的属性也可以修改具体的参数 |
3 func(void) | 函数指针,一个指向函数的指针。指向创建线程所执行函数的入口地址,函数执行完毕,则线程结束 |
4 arg | 线程主函数执行期间所使用的参数 |
运行实例
#include<stdio.h>
#include<pthread.h>
#include<unistd.h>
void *myfun()
{
printf("child pthid=%lu\n",pthread_self());
}
int main()
{
pthread_t pthid;
pthread_create(&pthid,NULL,myfun,NULL);
printf("parent pthid =%lu\n",pthread_self());
int i=0;
for(i;i<4;i++)
{
printf("i=%d\n",i);
}
sleep(2);
return 0;
}
2.终止线程——pthread_exit
void pthread_exit(void *rval_ptr)
参数是指针类型,用于存储线程结束后返回状态。
3.阻塞线程——pthread_join
int pthread_join(pthread_t thread, void ** rval_ptr)
调用该函数的线程将阻塞,直到thread线程调用pthread_exit、从启动例程返回或被取消,rval_ptr将包含返回码。即以阻塞的方式等待thread指定的线程结束。
注意:
调用该函数的线程将挂起等待,直到id为thread的线程终止。thread线程以不同的方法终止,通过pthread_join得到的终止状态是不同的,总结如下:
- 如果thread线程通过return返回, rval_ptr所指向的单元里存放的是thread线程函数的返回值。
- 如果thread线程被别的线程调用pthread_cancel异常终止掉, rval_ptr所指向的单元里存放的是常数PTHREAD_CANCELED。
- 如果thread线程是自己调用pthread_exit终止的,retval所指向的单元存放的是传给pthread_exit的参数。
- 如果对thread线程的终止状态不感兴趣,可以传NULL给rval_ptr参数。