学习嵌入式第二十九天

进程和线程(续)

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函数族

  1. 概念

    • 利用进程空间执行另一份代码

    • 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:执行新代码时更新环境变量
      
  2. 主函数传参

    • 主函数形式

      void main(void);
      int main(void);
      int main(int argc,const char *argv[]);
      int main(int argc,const char **argv);
      
      argc:传入参数的个数
      argv:存放每一个传入参数指针的数组名
      
  3. system函数

    原型:int system(const char *command);
    功能:运行command命令
    参数:
    	command:shell命令字符串首地址
    返回值:
    	成功返回0
    	失败返回-1
    

2.线程

1.概念

  • 是一个轻量级的进程
  • 进程空间和线程空间管理方法不同

2.进程和线程的区别

  1. 线程的本质是进程,线程是任务创建、调度、回收的过程
  2. 进程空间:文本段、数据段、系统数据段
  3. 线程空间:
    • 线程必须位于进程空间内部,没有进程,线程无法独立存在
    • 一个进程中的所有线程共享文本段、数据段、堆区,独享栈区
    • 独享的栈区空间大小默认8M
    • 一个进程中的多个线程切换调度任务时,资源开销较小

总结:线程是CPU任务调度的最小单元,进程是操作系统资源分配的最小单元

3.多进程多线程的优缺点

  • 对比

    场景多进程多线程对比
    效率多进程切换需要重新映射物理地址,占用资源开销较大多线程在同一进程空间内部切换任务,资源开销较小多线程>多进程
    通信多进程没有共享空间,需要使用进程间通信的方法来完成通信多线程有共享空间,只需定义共享空间变量完成数据交换即可实现通信多线程>多进程
    资源竞争多进程没有共享空间,不存在资源竞争多线程使用共享空间通信,需保证资源使用的互斥性,防止多线程对共享资源产生竞争多进程>多线程
    安全性多进程空间独立,一个进程的崩溃不会影响其余进程多线程公用空间,一个线程异常崩溃,可能引发进程异常退出,导致其余线程也无法执行多进程>多线程

4.线程的调度

  • 与进程调度保持一致
  • 宏观并行,微观串行

5.线程的消亡

  • 线程结束需要回收线程空间,否则产生僵尸线程

6.线程的函数接口

  • 进程接口线程接口
    forkpthread_create
    exitpthread_exit
    waitpthread_join
  1. 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
    	失败返回错误码
    
  2. pthread_self

    原型:pthread_t pthread_self(void);
    功能:获得调用该函数的线程的ID号
    
  3. pthread_exit

    原型:void pthread_exit(void *retval);
    功能:结束当前线程任务
    参数:
    	retval:线程结束的值
    
  4. 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;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值