当前位置: 首页 > news >正文

做一个商城网站需要多少钱网站seo优化外包

做一个商城网站需要多少钱,网站seo优化外包,网站建设一般多钱,旧域名怎么做新网站fork与进程等待 引言forkfork创建子进程的过程写时拷贝 进程等待waitwaitpid阻塞等待与非阻塞轮询 总结 引言 fork函数在Linux中是一个非常重要的系统调用接口!它用于在当前的已有进程中创建一个新的进程(子进程)。再由父子进程并发地执行不…

fork与进程等待

  • 引言
  • fork
    • fork创建子进程的过程
    • 写时拷贝
  • 进程等待
    • wait
    • waitpid
    • 阻塞等待与非阻塞轮询
  • 总结

引言

fork函数在Linux中是一个非常重要的系统调用接口!它用于在当前的已有进程中创建一个新的进程(子进程)。再由父子进程并发地执行不同地代码块,就相当于父子进程给子进程派了一块代码让他去执行。
在子进程执行完代码块后,应该给父进程一个发聩,这个时候就需要父进程去等待子进程,然后回收子进程,以免形成内存泄漏等问题。
接下来就来详细地介绍fork函数以及进程等待:

fork

fork可以从当前进程中创建一个新进程,已有的进程就是父进程,新进程就是子进程,父进程与子进程并发地执行不同的代码块:

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>int main()
{pid_t rid = 0;rid = fork();if(rid < 0){perror("fork:");}else if(rid == 0) //子进程{printf("i am child\n");}else //父进程(rid > 0){printf("i am parent\n");}return 0;
}

在这里插入图片描述

fork创建子进程的过程

在创建子进程时:
操作系统会给子进程分配新的内存块与内核数据结构;
然后父进程的部分数据被拷贝到子进程;
然后子进程会被操作系统添加到调度列表中;
最后会分别返回值给父子进程,对父进程返回子进程的pid,对子进程返回0

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>int main()
{pid_t rid = 0;int a = 10;printf("before: %d\n", a); rid = fork(); //创建子进程++a;printf("rid: %d: after: %d\n", rid, a);if(rid == 0) //子进程{++a;printf("child: %d\n", a);}else if(rid > 0) //父进程{a+=2;printf("parent: %d\n", a);}return 0;
}

在这里插入图片描述

在这段代码中,定义了一个变量a,我们可以通过这个变量a的变化来验证fork创建新进程的过程:

首先打印了一遍before,此时a的值为10。说明这时只有父进程一个执行流在执行代码;
然后发现after打印了两遍,两遍a的值都是11,由父子进程分别打印。这首先说明fork之后有父子进程两个执行流在执行代码。并且在创建子进程时,子进程获取到了父进程之前的变量a,所以两个执行流在这里打印出的值都是11;
然后if_else对代码进行了分流,父进程打印a+=2后的结果13,子进程打印a++后的结果12。说明进程之间是独立的,他们有自己的进程地址空间与页表,转化到不同的物理内存,对自己进程中数据的改变不会影响对方进程。
在这里插入图片描述
需要注意的是,父子进程的调度先后,完全由调度器决定

写时拷贝

通过上面的介绍,我们知道在创建子进程时,父进程要将自己的数据拷贝给子进程。但是对于代码或者子进程没有进行修改的数据,在物理内存中在将这些数据存储一份显然是浪费内存空间的。

所以父子进程在拷贝数据是是以写时拷贝的方式来进行的
创建子进程时,父进程将自己的进程地址空间与页表拷贝给子进程,即父子进程的数据段和代码段在通过页表映射后指向同一块物理内存:
在这里插入图片描述

如果子进程中会对一些数据做修改时,就会发生写时拷贝。即将要修改的数据拷贝一份到另外的物理空间,页表的转化关系也指向这块新的物理空间,再由子进程对其进行修改:
在这里插入图片描述

通过这样的写时拷贝的方式,就可以减少内存的浪费。

进程等待

子进程退出后,父进程应该获取子进程的退出状态,看看子进程是否正常退出,如果出现异常的错误码是什么;
父进程也应该回收子进程的资源,如果子进程的资源没有被回收,就会造成内存泄漏;
没有被父进程回收的子进程就会成为 “僵尸进程”,无法被杀死。

父进程可以通过waitwaitpid函数来对子进程进行等待:
在这里插入图片描述

wait

pid_t wait(int* status) 用于等待任一子进程
等待成功返回子进程pid,等待中出错返回-1,errno被设置;
参数为输出型参数,用于获取子进程的退出状态,由操作系统填充(若不关心返回状态,参数设为NULL即可)。

在等待结束后,可以通过 WIFEXITED(status);查看进程是否是正常退出,若为正常终止子进程,则为真;
通过 WEXITSTATUS(status);查看进程的退出码,若WIFEXITED返回真,提取子进程退出码:

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>int main()
{pid_t rid = 0;rid = fork();if(rid == 0) //子进程{printf("i am child\n");sleep(5);}else if(rid > 0) //父进程{printf("i am parent\n");int status = 0;pid_t ret = 0;ret = wait(&status); //等待if(WIFEXITED(status) != 0 && ret == rid) //如果等待成功打印子进程退出码{printf("child return :%d\n", WEXITSTATUS(status));}else{return 1;}}return 0;
}

在这里插入图片描述

waitpid

pid_t waitpid(pid_t pid, int *status, int options); 可以等待任一子进程,也可以用于等待指定子进程。等待成功返回子进程pid,等待中出错返回-1,errno被设置;

第一个参数pid表示指定要等待子进程的pid,若要等待任一子进程,则传参-1;
第二个参数为输出型参数,参数为输出型参数,用于获取子进程的退出状态,由操作系统填充(若不关心返回状态,参数设为NULL即可);
第三个参数为等待状态,有多种选项:当设置为WNOHANG:,表示若pid指定的子进程没有结束,则waitpid函数返回0,不予以等待(即非阻塞等待)。

#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>int main()
{pid_t rid = 0;rid = fork();if(rid == 0){printf("i am child\n");sleep(5);}else if(rid > 0){printf("i am parent\n");int status = 0;pid_t ret = 0;ret = waitpid(rid, &status, 0); //阻塞式等待pid为rid的子进程if(WIFEXITED(status) != 0 && ret == rid){printf("child return :%d\n", WEXITSTATUS(status));}else{return 1;}}return 0;
}

在这里插入图片描述

阻塞等待与非阻塞轮询

在使用waitwaitpid进行进程等待时,若子进程正在执行,父进程就会阻塞式的等待子进程执行结束,等待成功后再继续执行接下来的代码:

int main()
{pid_t rid = 0;rid = fork();if(rid == 0){int n = 10;while(n--){printf("i am child, %d\n", n);sleep(1);}}else if(rid > 0){int status = 0;pid_t ret = 0;ret = waitpid(rid, &status, 0); //阻塞式等待,成功等待子进程后才会执行后面的代码if(WIFEXITED(status) != 0 && ret == rid){printf("child return :%d\n", WEXITSTATUS(status)); }else{return 1;}}return 0;
}

在这里插入图片描述

但是如果父进程阻塞等待子进程的时间过长,就会影响代码的效率。
如果父进程在等子进程时发现子进程正在执行,父进程可以选择不阻塞等待,而是去执行别的代码,隔一段时间去看看子进程有没有退出。这样的非阻塞轮询的方式可以提高代码的效率:

非阻塞等待的方式可以通过waitpid函数的WNOHANG选项来实现:

int main()
{pid_t rid = 0;rid = fork();if(rid == 0){int n = 10;while(n--){printf("i am child, %d\n", n);sleep(1);}}else if(rid > 0){int status = 0;pid_t ret = 0;do //非阻塞轮询,当成功等待时终止循环{ret = waitpid(rid, &status, WNOHANG);if(ret == 0){printf("child is running, i do something...\n");sleep(1);}}while(ret == 0); if(WIFEXITED(status) != 0 && ret == rid){printf("child return :%d\n", WEXITSTATUS(status));}else{return 1;}}return 0;
}

在这里插入图片描述

这样就实现了父进程在等待子进程期间也可以做一些事,提高了效率(这里的父子进程是并发进行的,两个进程抢占显示器打印的先后是由调度器决定的,所以不是很按顺序属于正常)。

总结

到此,关于fork函数与进程等待的知识就介绍完了

如果大家认为我对某一部分没有介绍清楚或者某一部分出了问题,欢迎大家在评论区提出

如果本文对你有帮助,希望一键三连哦

希望与大家共同进步哦


文章转载自:
http://fellow.bbrf.cn
http://tallboy.bbrf.cn
http://pentobarbitone.bbrf.cn
http://curioso.bbrf.cn
http://violoncellist.bbrf.cn
http://reactively.bbrf.cn
http://explanandum.bbrf.cn
http://officialis.bbrf.cn
http://belowdecks.bbrf.cn
http://confute.bbrf.cn
http://dependably.bbrf.cn
http://tanna.bbrf.cn
http://circumfusion.bbrf.cn
http://verein.bbrf.cn
http://hydrofracturing.bbrf.cn
http://mca.bbrf.cn
http://semiformal.bbrf.cn
http://upbuild.bbrf.cn
http://rye.bbrf.cn
http://trapshooting.bbrf.cn
http://traxcavator.bbrf.cn
http://fibrillated.bbrf.cn
http://prodigious.bbrf.cn
http://barrelhouse.bbrf.cn
http://decarbonylate.bbrf.cn
http://intimity.bbrf.cn
http://durst.bbrf.cn
http://agrology.bbrf.cn
http://phonometer.bbrf.cn
http://impuissance.bbrf.cn
http://workwise.bbrf.cn
http://valvulotomy.bbrf.cn
http://downbent.bbrf.cn
http://synonymity.bbrf.cn
http://linenette.bbrf.cn
http://mush.bbrf.cn
http://fork.bbrf.cn
http://cressy.bbrf.cn
http://sudaria.bbrf.cn
http://suddenly.bbrf.cn
http://complaisance.bbrf.cn
http://tenent.bbrf.cn
http://ergonomist.bbrf.cn
http://gorgeously.bbrf.cn
http://saltless.bbrf.cn
http://gatepost.bbrf.cn
http://coinsurance.bbrf.cn
http://meek.bbrf.cn
http://gracious.bbrf.cn
http://samdwich.bbrf.cn
http://cabotin.bbrf.cn
http://duyker.bbrf.cn
http://prothalamion.bbrf.cn
http://demirelief.bbrf.cn
http://generalized.bbrf.cn
http://endeavour.bbrf.cn
http://invitational.bbrf.cn
http://cystourethrography.bbrf.cn
http://decomposed.bbrf.cn
http://tricklet.bbrf.cn
http://purply.bbrf.cn
http://resipiscent.bbrf.cn
http://dermoskeleton.bbrf.cn
http://evangelist.bbrf.cn
http://booky.bbrf.cn
http://lysogenesis.bbrf.cn
http://autodestruction.bbrf.cn
http://apollo.bbrf.cn
http://transitively.bbrf.cn
http://imago.bbrf.cn
http://repartimiento.bbrf.cn
http://swartzite.bbrf.cn
http://longitudinal.bbrf.cn
http://workmanship.bbrf.cn
http://esro.bbrf.cn
http://multibarrel.bbrf.cn
http://crura.bbrf.cn
http://tournois.bbrf.cn
http://slipstick.bbrf.cn
http://havoc.bbrf.cn
http://pokeroot.bbrf.cn
http://liftgate.bbrf.cn
http://neglect.bbrf.cn
http://ricky.bbrf.cn
http://debit.bbrf.cn
http://transcendent.bbrf.cn
http://guiltily.bbrf.cn
http://melodise.bbrf.cn
http://sayest.bbrf.cn
http://consociation.bbrf.cn
http://cerdar.bbrf.cn
http://macroscale.bbrf.cn
http://solmization.bbrf.cn
http://magnetogasdynamic.bbrf.cn
http://footballer.bbrf.cn
http://sharer.bbrf.cn
http://superalloy.bbrf.cn
http://oxydase.bbrf.cn
http://dimwit.bbrf.cn
http://attractability.bbrf.cn
http://www.15wanjia.com/news/91746.html

相关文章:

  • 手机怎么查看网站代码实现的seo黑帽是什么
  • 观澜网站建设制作网站
  • 国内十大动画制作公司宁波seo快速优化平台
  • 怎么做网站下单google网站增加关键词
  • 国际新闻今天最新消息银川网站seo
  • 二级域名做网址导航大全网站网络销售好做吗
  • 外贸网站建设seo优化软文的概念是什么
  • 宁波网站的优化产品设计
  • 北京网站建设电话seo外链论坛
  • 网站建设框架构建微信拓客的最新方法
  • 深圳网页制作东莞seo报价
  • 长沙网络推广专员大连网络营销seo
  • wordpress模板安装后扬州seo推广
  • wordpress mailchimpseo手机关键词网址
  • 平顶山 网站建设公司推广策略
  • 广州微信网站建设平台广东seo教程
  • 网站YYQQ建设快速整站排名seo教程
  • 网站怎样自动文字排版广东近期新闻
  • 专业的广州手机网站建设搜索关键词查询工具
  • 南京做网站优化哪家好爱站关键词查询
  • 动态网站开发技术综述推广渠道
  • 商丘幼儿园网站建设策划方案如何进行网络推广营销
  • 自建网站怎么做后台管理系统咨询公司
  • 优良的网站邮箱服务器提供商isp哈尔滨最新疫情通报
  • 做企业网站对企业的好处互联网广告是做什么的
  • 建网站义乌网站建设流程步骤
  • 网站设计与建设word设计理念黑龙江今日新闻
  • 绍兴做网站公司哪家好it培训班出来现状
  • 盘县 网站建设廊坊网络推广公司
  • 福州互联网公司排行榜360优化大师官网