进程和线程(续)
1.进程
-
waitpid
原型:pid_t waitpid(pid_t pid, int *wstatus, int options); 功能:回收指定的子进程空间 参数: pid:要回收的进程的PID(-1表示回收任一子进程) wstatus:存放子进程结束状态空间首地址 options: 0 阻塞回收 WHONANG 非阻塞回收 返回值: 成功返回回收到的子进程PID 失败返回-1 指定的子进程空间没结束,返回0
- waitpid可以非阻塞回收子进程空间
- wait可以回收指定子进程空间
exec函数族
-
概念
-
利用进程空间执行另一份代码
-
exec常搭配fork使用,fork负责创建子进程,exec负责让子进程执行自己的代码
extern char **environ; int execl(const char *path, const char *arg, .../* (char *) NULL */); int execlp(const char *file, const char *arg, .../* (char *) NULL */); int execle(const char *path, const char *arg, .../*, (char *) NULL, char * const envp[] */); int execv(const char *path, char *const argv[]); int execvp(const char *file, char *const argv[]); int execvpe(const char *file, char *const argv[],char *const envp[]); l:参数以列表的形式传递 p:在系统变量PATH对应的目录下查找文件 v:参数以指针数组的形式传递 e:执行新代码时更新环境变量
-
-
主函数传参
-
主函数形式
void main(void); int main(void); int main(int argc,const char *argv[]); int main(int argc,const char **argv); argc:传入参数的个数 argv:存放每一个传入参数指针的数组名
-
-
system函数
原型:int system(const char *command); 功能:运行command命令 参数: command:shell命令字符串首地址 返回值: 成功返回0 失败返回-1
2.线程
1.概念
- 是一个轻量级的进程
- 进程空间和线程空间管理方法不同
2.进程和线程的区别
- 线程的本质是进程,线程是任务创建、调度、回收的过程
- 进程空间:文本段、数据段、系统数据段
- 线程空间:
- 线程必须位于进程空间内部,没有进程,线程无法独立存在
- 一个进程中的所有线程共享文本段、数据段、堆区,独享栈区
- 独享的栈区空间大小默认8M
- 一个进程中的多个线程切换调度任务时,资源开销较小
总结:线程是CPU任务调度的最小单元,进程是操作系统资源分配的最小单元
3.多进程多线程的优缺点
-
对比
场景 多进程 多线程 对比 效率 多进程切换需要重新映射物理地址,占用资源开销较大 多线程在同一进程空间内部切换任务,资源开销较小 多线程>多进程 通信 多进程没有共享空间,需要使用进程间通信的方法来完成通信 多线程有共享空间,只需定义共享空间变量完成数据交换即可实现通信 多线程>多进程 资源竞争 多进程没有共享空间,不存在资源竞争 多线程使用共享空间通信,需保证资源使用的互斥性,防止多线程对共享资源产生竞争 多进程>多线程 安全性 多进程空间独立,一个进程的崩溃不会影响其余进程 多线程公用空间,一个线程异常崩溃,可能引发进程异常退出,导致其余线程也无法执行 多进程>多线程
4.线程的调度
- 与进程调度保持一致
- 宏观并行,微观串行
5.线程的消亡
- 线程结束需要回收线程空间,否则产生僵尸线程
6.线程的函数接口
-
进程接口 线程接口 fork pthread_create exit pthread_exit wait pthread_join
-
pthread_create
原型:int pthread_create(pthread_t *thread, const pthread_attr_t *attr,void *(*start_routine) (void *), void *arg); 功能:在进程中创建一个线程 参数: thread:存放线程ID空间的首地址 attr:线程的属性,默认属性NULL start_routine:线程函数的入口 arg:线程传入的参数 返回值: 成功返回0 失败返回错误码
-
pthread_self
原型:pthread_t pthread_self(void); 功能:获得调用该函数的线程的ID号
-
pthread_exit
原型:void pthread_exit(void *retval); 功能:结束当前线程任务 参数: retval:线程结束的值
-
pthread_join
原型:int pthread_join(pthread_t thread, void **retval); 功能:回收线程空间 参数: thread:要回收的线程ID retval:存放线程结束状态空间的首地址 返回值: 成功返回0 失败返回错误码
- tid对应的线程只要不退出,pthread_join阻塞等待结束回收线程空间
- pthread_join具备同步功能
习题
1.找到并打印出指定目录下的音视频文件
代码实现:
#include <stdio.h>
#include <string.h>
#include <dirent.h>
#include <sys/types.h>
int get_mv(char *file){
int i = 0;
while(*file != '\0'){
file++;
i++;
}
while(i > 0){
if(*file == '.'){
break;
}
file--;
i--;
}
if(i == 0){
return 0;
}
if(strcmp(file,".mp4") == 0 || strcmp(file,".avi") == 0 || strcmp(file,".rmvb") == 0 || strcmp(file,".mp3") == 0 || strcmp(file,".flv") == 0 || strcmp(file,".wma") == 0 || strcmp(file,".txt") == 0){
return 1;
}
return 0;
}
int main(){
DIR *dir = NULL;
struct dirent *ptr = NULL;
dir = opendir("mv");
if(dir == NULL){
perror("fail to opendir");
return -1;
}
while(1){
ptr = readdir(dir);
if(ptr == NULL){
break;
}
if(get_mv(ptr->d_name) == 1){
printf("file = %s\n", ptr->d_name);
}
}
closedir(dir);
return 0;
}
2.创建4个线程任务:
- 线程1间隔1s打印“采集线程正在执行”
- 线程2间隔2s打印“存储线程正在执行”
- 线程3间隔5s打印“显示线程正在执行”
- 线程4间隔10s打印“日志线程正在执行”
代码实现:
#include <stdio.h>
#include <string.h>
#include <dirent.h>
#include <sys/types.h>
#include <pthread.h>
#include <unistd.h>
void *thread1(void *arg){
while(1){
sleep(1);
printf("采集线程正在执行\n");
}
}
void *thread2(void *arg){
while(1){
sleep(2);
printf("存储线程正在执行\n");
}
}
void *thread3(void *arg){
while(1){
sleep(5);
printf("显示线程正在执行\n");
}
}
void *thread4(void *arg){
while(1){
sleep(10);
printf("日志线程正在执行\n");
}
}
int main(void){
pthread_t tid[4];
int i = 0;
void *(*p[4])(void *) = {thread1, thread2, thread3, thread4};
for(i = 0; i < 4; i++){
pthread_create(&tid[i],NULL,p[i],NULL);
}
while(1){
}
return 0;
}