• C语言:多进程的详细介绍


    1.多进程相关概念

    1.1什么是进程?

    进程:程序的一次执行过程就会产生一个进程。进程是分配资源的最小单位(0-3G)。

    进程就是一个正在执行的任务。进程是一个动态的过程,它有生命周期随着程序的运行

    开始,随着程序结束而消亡。每个进程都有自己的独立的运行的空间,比如每个进程都有

    自己的文件描述符,每个进程都拥有自己的缓冲区。只要用户执行了一个程序,在内核空间

    就会创建一个task_struct的结构体,这个结构体就代表当前的进程。进程运行产生的所有

    的信息都被放到这个结构体中保存着。

    1.2进程和程序有什么区别?

    程序:程序是经过编译器编译生成的二进制文件,程序在硬盘上存储。程序是静态的,没有生命

    ​ 周期的概念程序本身不会分配内存。

    进程:程序的一次执行过程就会创建一个进程,进程是动态的,有生命周期。进程运行的时候会

    ​ 分配0-3G的内存空间。进程在内存上存储。

    1.3进程的组成?

    进程是由三个部分组成的:进程的PCB(task_struct),(程序段)文本段,数据段。

    1.4进程的种类?

    交互进程:这种进程维护以一个终端,通过这个终端用户可以和进程进程交互。

    ​ 例如:文本编辑器

    批处理进程:这种进程优先级比较低,运行的时候会被放到一个运行的队列中。

    ​ 随着队列的执行,而逐渐执行。

    ​ 例如gcc编译程序的时候这个进程就是批处理进程。

    守护进程:守护进程就是后台运行的服务,随着系统的启动而启动,随着系统的终止而终止。

    ​ 例如:windows上的各种服务

    1.5什么是进程的PID

    PID就是操作系统给进程分配的编号,它是识别进程的唯一的标识。

    在linux系统中PID是一个大于等于0的值。

    1.6特殊PID的进程

    0:idle:在linux系统启动的时候运行的第一个进程就是0号进程。

    ​ 如果没有其他的进程执行就运行这个idle进程。

    1:init:1号进程是由0号进程调用内核kernel_thread函数产生的第一个进程,

    ​ 它会初始化所有的系统的硬件。当初始化完之后会一直执行,比如会为

    ​ 孤儿进程回收资源。

    2:kthreadd:调度器进程,主要负责进程的调度工作。

    1.7进程的状态

     1.进程运行的状态
     D    // 不可中断的休眠态(信号)
        R    //运行态
        S    //可中断的休眠态(信号)
        T    //停止状态
        X    //死亡状态
        Z    //僵尸态(进程结束后没有被父进程回收资源)
    2.进程的附加状态
       <    //高优先级
       N    //低优先级的进程
       L    //在内存区锁定
       s    //会话组组长
       l    //进中包含多线程
       +    //前台进程

    二.进程实例:

    2.1进程的拷贝:

    1. #include
    2. int main(int argc,const char * argv[])
    3. {
    4. for(int i=0;i<2;i++){
    5. fork();
    6. printf("-");
    7. }
    8. return 0;
    9. }

    原理图:

     

    分析:上述的程序会打印8个-,上述程序一共是4个进程,因为在进程创建的时候会拷贝父进程的

    缓冲区,由于2214和2215的缓冲区没有刷新,所以2216和2217缓冲区中有两个-。

    2.2进程创建的实例

    1. #include
    2. int main(int argc,const char * argv[])
    3. {
    4. pid_t pid;
    5. pid = fork();
    6. if(pid == -1){
    7. PRINT_ERR("fork error");
    8. }else if(pid == 0){
    9. //子进程
    10. printf("这个就是子进程\n");
    11. }else{
    12. //父进程
    13. printf("这个就是父进程\n");
    14. }
    15. return 0;
    16. }

    分析图:

     2.3执行没有先后循序:

    1. #include
    2. int main(int argc, const char* argv[])
    3. {
    4. pid_t pid;
    5. pid = fork();
    6. if (pid == -1) {
    7. PRINT_ERR("fork error");
    8. } else if (pid == 0) {
    9. //子进程
    10. while (1) {
    11. sleep(1);
    12. printf("这个就是子进程\n");
    13. }
    14. } else {
    15. //父进程
    16. while (1) {
    17. sleep(1);
    18. printf("这个就是父进程\n");
    19. }
    20. }
    21. return 0;
    22. }

    结果图:
     

     

    2.4进程创建的实例(父子进程内存空间问题)

    父进程fork产生子进程的时候利用了写时拷贝的原则

     

    2.5使用多进程创建三个进程:

    1. #include
    2. #include
    3. #include
    4. #include
    5. int main(int argc, char const *argv[])
    6. {
    7. pid_t pid;
    8. if((pid = fork()) ==-1){
    9. printf("创建进程失败\n");
    10. exit(-1);
    11. }else if(pid == 0){
    12. pid_t tid;
    13. if((tid = fork())==-1){
    14. printf("%d,%d\n",getpid(),getppid());
    15. }else if(pid == 0){
    16. printf("%d,%d\n",getpid(),getppid());
    17. }else{
    18. printf("%d\n",getpid());
    19. }
    20. }
    21. while(1);
    22. return 0;
    23. }

    2.6使用多进程进行拷贝文件:

    1. #include
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. char destfile[32]={0};
    8. int get_length(char const *arcfile)
    9. {
    10. int fd,cd,length;
    11. if((fd=open(arcfile,O_RDONLY))==-1){
    12. printf("open file error");
    13. exit(-1);
    14. }
    15. snprintf(destfile,sizeof(destfile),"new_%s",arcfile);
    16. if((cd==open(destfile,O_WRONLY|O_TRUNC|O_CREAT,0666)==-1)){
    17. }
    18. length = lseek(fd,0,SEEK_END);
    19. return length;
    20. }
    21. int copy_file(const char* srcfile,const char* destfile,int start,int length)
    22. {
    23. int sfd,dfd,ret,count = 0;
    24. char buff[128] = {0};
    25. int counts=0;
    26. if((sfd=open(srcfile,O_RDONLY))==-1){
    27. printf("open error");
    28. }
    29. if((dfd=open(destfile,O_WRONLY))==-1){
    30. printf("open file error");
    31. }
    32. lseek(sfd,start,SEEK_SET);
    33. lseek(dfd,start,SEEK_SET);
    34. while (1){
    35. ret = read(sfd,buff,sizeof(buff));
    36. if(ret == 0){
    37. break;
    38. }
    39. count+=ret;
    40. if(count >= length){
    41. write(dfd,buff,length-(count-ret));
    42. break;
    43. }
    44. write(dfd,buff,ret);
    45. }
    46. return 0;
    47. }
    48. int main(int argc, char const *argv[])
    49. {
    50. pid_t pid;
    51. if(argc!=2){
    52. printf("输入格式错误,请重新输入\n");
    53. exit(-1);
    54. }
    55. int length = get_length(argv[1]);
    56. if(length < 0){
    57. printf("srcfile error\n");
    58. return -1;
    59. }
    60. if((pid=fork())==-1){
    61. printf("fork error");
    62. exit(-1);
    63. }else if(pid == 0){
    64. copy_file(argv[1],destfile,length/2,length-length/2);
    65. }else{
    66. copy_file(argv[1],destfile,0,length/2);
    67. }
    68. return 0;
    69. }

    三.什么是孤儿进程,什么是僵尸进程

    3.1僵尸进程:

    子进程结束之后,父进程没有为它回收资源,此时子进程就是僵尸进程

    1. #include
    2. int main(int argc, const char* argv[])
    3. {
    4. pid_t pid;
    5. pid = fork();
    6. if (pid == -1) {
    7. PRINT_ERR("fork error");
    8. } else if (pid == 0) {
    9. //子进程
    10. } else {
    11. //父进程
    12. while(1);
    13. }
    14. return 0;
    15. }

    3.2孤儿进程:

    孤儿进程:父进程死掉之后,子进程就是孤儿进程,孤儿进程被init进程收养。

    1. #include
    2. int main(int argc, const char* argv[])
    3. {
    4. pid_t pid;
    5. pid = fork();
    6. if (pid == -1) {
    7. PRINT_ERR("fork error");
    8. } else if (pid == 0) {
    9. //子进程
    10. printf("pid = %d\n",getpid());
    11. while (1);
    12. } else {
    13. //父进程
    14. }
    15. return 0;
    16. }

    四,守护进程的创建:

    4.1守护进程的创建

    守护进程:相当于系统的服务,随着系统的启动而运行,随着系统的终止而终止。脱离了某个终端.

    4.2步骤图:

     

    4.3代码实现:

    1. #include
    2. #include
    3. #include
    4. #include
    5. #include
    6. #include
    7. #define ERROR(msg) do{\
    8. printf("%s %s %d\n",__FILE__,__func__,__LINE__);\
    9. printf(msg);\
    10. exit(-1); \
    11. }while(0)
    12. int main(int argc, char const *argv[])
    13. {
    14. pid_t pid;
    15. if ((pid == fork()) == -1){
    16. ERROR("fork error");
    17. }else if(pid == 0){
    18. int fd;
    19. setsid();
    20. chdir("/");
    21. umask(0);
    22. for(int i = 3; i < getdtablesize(); i++){
    23. close(i);
    24. }
    25. if((fd = open("my.log", O_RDWR, O_APPEND, O_CREAT, 0666)) == -1){
    26. ERROR("open error");
    27. }
    28. dup2(fd, 0);
    29. dup2(fd, 1);
    30. dup2(fd, 2);
    31. while(1){
    32. sleep(1);
    33. printf("我是子进程\n");
    34. fflush(stdout);
    35. }
    36. }else{
    37. printf("父亲进程已经结束....\n");
    38. exit(-1);
    39. }
    40. return 0;
    41. }

  • 相关阅读:
    用文字描述给黑白照上色,这个免费网站火了!网友:比其他同类都好用
    VM装MACos
    .net core 的 winform 的 浏览器控件 WebView2
    Baichuan2:Open large-scale language models
    前端研习录(36)——ES6 字符串扩展及新增方法讲解及示例分析
    数组排序,实现中间数字最大,向两边越来越小的排序
    【ES6知识】ESModule 模块化
    TypeScript 实践中的 Equals 类型的实现原理是什么?
    浅讲make/makefile【linux】
    java: 警告: 源发行版 17 需要目标发行版 17
  • 原文地址:https://blog.csdn.net/a2998658795/article/details/126324352