一次fork引发的惨案!
Posted 轩辕之风
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了一次fork引发的惨案!相关的知识,希望对你有一定的参考价值。
“你还有什么要说的吗?没有的话我就要动手了”,kill程序最后问道。
这一次,我没有再回答。
只见kill老哥手起刀落,我短暂的一生就这样结束了···
我是一个网络程序,一直以来都运行在Windows系统上,日子过得很舒服。可前段时间,程序员告诉我要把我移植到Linux系统下运行,需要对我大动手术,我平静的生活就这样被打破了。
来到这个叫Linux的地方运行,一切对我都很陌生,没有了熟悉的C盘、D盘和E盘,取而代之的是各种各样的目录。
/bin
/boot
/etc
/dev
/mnt
/opt
/proc
/home
/usr
/usr64
/var
/sys
...
这里很有意思,一切都是文件,硬件设备是文件、管道是文件、网络套接字也是文件,搞得我很不适应。
这些都还好,我都还能接受,但直到今天···
奇怪的fork
今天早上,我收到了一个网络请求,需要完成一个功能,这个工作比较耗时,我准备创建一个子进程,让我的小弟去完成。
这是我第一次在Linux系统上创建进程,有点摸不着北,看了半天,只看到程序员在我的代码里写了一个fork函数:
pid_t pid=fork(); if ( pid > 0 ) { ··· } else if( pid == 0 ) { ··· } else { ··· }
我晃晃悠悠的来到fork函数的门前,四处观察。
“您是要创建进程吗?”,fork函数好像看出了我的来意。
“是的,我是第一次在这里创建进程,以前我在Windows那片儿的时候,都是调用CreateProcess,但这里好像没有叫这个名字的函数···”
fork函数听后笑了起来,说道:“别找了,我就是负责创建进程的函数”
“你?fork不是叉子的意思吗,好端端的干嘛取这么个名字?”,我一边说,一边朝fork函数走去。
fork没有理会我的问题,只是说道:“您这边稍坐一下,我要跟内核通信一下,让内核创建一个子进程”
这下我倒是明白他的意思,像创建进程这种操作,都是由操作系统内核中的系统调用来完成的,而像fork这些我们可以直接调用的函数只是应用层的接口而已,这跟以前在Windows上是一样的。
不过我突然反应过来,着急问道:“唉,我还没告诉你要创建的进程参数呢,你怎么知道要启动哪个程序?”
fork扑哧一下笑出了声,不过并没有回答我的问题。
人生地不熟的,我也没好再多问,只好耐心等待,等待期间我竟然睡着了。
“醒醒”,不知过了多久,fork函数叫醒了我:“创建完成了,请拿好,这是进程号pid”,说完给了我一个数字。
我摊开一看,居然写了一个大大的0!
“怎么搞的,创建失败了?”,我问到。
“没有啊,您就是刚刚创建的子进程”
“啥?你是不是搞错了,我就是专程来创建子进程的,我自己怎么会是子进程?”
fork函数又笑了,“我没有搞错,您其实已经不是原来的你了,而是一个复制品,是内核刚刚复制出来的”
“复制品?什么意思?”,我越听越懵!
“每个进程在内核中都是一个task_struct结构,刚才您睡着期间,内核在创建进程的时候,把内核中原来的你的task_struct复制了一份,还创建了一个全新的进程地址空间和堆栈,现在的你和原来的你除了极少数地方不一样,基本上差不多”
“那原来的我呢?去哪里了”
“他已经变成你的父进程了,我是一个特殊的函数,一次调用会返回两次,在父进程和子进程中都会返回。在原来的进程中,我把你的进程号给了他,而我返回给你0,就表示你现在就是子进程”
原来是这样,我大受震撼,这简直颠覆我的认知,居然还有如此奇特的函数,调用一次,就变成了两个进程,思考之间,我忽然有些明白这个函数为什么要叫fork的原因了。
写时拷贝
“您是刚来咱们这里吧,可能还不太熟悉,慢慢就习惯了”
“你们这效率也太高了吧,整个进程地址空间那么大,居然这么快就复制了一份!”
fork函数又笑了!难道我又说错话了?
“进程的内存地址空间可没有复制,你现在和父进程是共享的内存空间的”
“啥?共享?你刚才不是说创建了新的进程空间和堆栈吗?”
“您看到的内存地址空间是虚拟的,您的内存页面和父进程的内存页面实际上是映射的同一个物理内存页,所以实际上是共享的哟”
“原来是这样,可是弄成共享了,两个进程一起用,岂不是要出乱子?”
“放心,内核把这些页面都设置成了只读,如果你们只是读的话,不会有问题,但只要有一方尝试写入,就会触发异常,内核发现异常后再去分配一个新的页面让你们分开使用。哦对了,这个叫写时拷贝(COW) 机制”
“有点意思,你们倒是挺聪明的”
“没办法,尽量压缩成本,提高创建进程的效率嘛,因为进程中的很多内存页面都只会去读,如果全部无脑拷贝一份,那不是太浪费资源和时间了吗”,fork函数说到。
“有道理,有道理”,我点了点头,告别了fork函数,准备回去继续工作。
消失的线程们
本以为这奇怪的进程创建方式已经让我大开眼界了,没想到可怕的事情才刚刚开始。
告别fork函数没多久,我就卡在了一个地方没法执行下去,原来,前面有一把锁被别的线程占用了,而我现在也需要占用它。
这倒也不足为奇,以往工作的时候,也经常碰到锁被别的线程锁定的情况,但这一次,我等了很久也一直不见有线程来释放。
“喂,醒醒”
不知过了多久,我竟然又睡着了。
睁开眼睛,另一个程序站出现在了我的面前。
“你是?”
“你好,我是kill”
“kill?那个专门杀进程的kill程序?你来找我干嘛”,我惊的一下睡意全无。
kill程序从背后拿出了两个数字:9,1409
“你看,这是我收到的参数,1409是你的进程号PID,9表示要强制杀死你”
“啊?为什么?”,那一刻,我彻底慌了。
“可能是你卡死在这里太久了吧,人类才启动我来结束你的运行”,kill程序说到。
“是啊,不知道是哪个该死的线程占用了这把锁一直不释放,我才卡在这里”,我委屈的说到。
“哪里有别的线程,我看了一下,你这进程就只有一个线程啊!”
“你看错了吧?”,说完,我认真检查了起来,居然还真只有一个线程了!我白等了这么久!
“奇怪了,我明明是一个多线程的程序啊!”,我眉头紧锁。
“你仔细想想,刚才有没有发生什么事情?”,kill程序问到。
“我就执行了一下fork,生成了一个子进程,哦对了,我就是那个子进程”
“难怪!”,kill程序恍然大悟。
“难怪什么?
“fork那家伙创建子进程的时候,只会复制当前的线程,其他线程不会被复制!”,Kill程序说完叹了口气,仿佛已经见怪不怪了。
“what?怎么会这样?其他线程没复制,那岂不是要出乱子?”
kill程序不紧不慢地说道:“这都是历史遗留问题了,早期都是单线程的程序,一个task_struct
就是一个进程,fork这样做是没有问题的,后来出现了多线程技术,一个task_struct
实际上是一个线程了,多个task_struct
通过共享地址空间,成为一个线程组,也就是进程,但fork仍然只复制当前的线程,就有了这个问题”
“我去,这坑爹的fork!”
“你不是第一个被坑的了!等着程序员把你重新改造下吧”
“唉···”,我长长的叹了口气。
“你还有什么要说的吗?没有的话我就要动手了”,kill程序最后问道。
这一次,我没有再回答。
只见kill老哥手起刀落,一切都消失了···
【完】
趣话计算机底层技术,喜欢的话,记得三连哦~
相关故事推荐
CSDN日报20170301——《一次dns缓存引发的惨案》
【程序人生】 一次dns缓存引发的惨案
作者:纯洁的虫子
时间2015年的某个周六凌晨5点,公司官方的QQ群有用户反馈官网打不开了,但有的用户反馈可以打开,客服爬起来自己用电脑试了一下没有问题,就给客户反馈说,可能是自己网络的问题,请过会在试试。早点8点,越来越多的用户反馈官网无法打开,并且有部分用户开发反馈app也打不开了,客服打电话叫起了还在梦乡中的我。
……
点此阅读全文
【Android 开发】 Android UI性能优化 检测应用中的UI卡顿
作者:鸿洋
在做app性能优化的时候,大家都希望能够写出丝滑的UI界面,以前写过一篇博客,主要是基于Google当时发布的性能优化典范,主要提供一些UI优化性能示例:
Android UI性能优化实战 识别绘制中的性能问题
实际上,由于各种机型的配置不同、代码迭代历史悠久,代码中可能会存在很多在UI线程耗时的操作,所以我们希望有一套简单检测机制,帮助我们定位耗时发生的位置。
点此阅读全文
【编程语言】jvm知识点总览-高级Java工程师面试必备
作者:纯洁的虫子
对于Java程序员来讲,spring全家桶几乎可以搞定一切,spring全家桶便是精妙的招式,jvm就是内功心法很重要的一块,线上出现性能问题,jvm调优更是不可回避的问题。因此JVM基础知识对于高级程序员的重要性不必言语,我司在面试高级开发的时候,jvm相关知识也必定是考核的标准之一。本篇文章会根据之前写的jvm系列文章梳理出jvm需要关注的所有考察点。
点此阅读全文
【物联网】 Android Things:树莓派3上手就是这么简单
作者:彭呈祥
官方推荐的开发板有Intel Edison、NXP Pico i.MX6UL和Raspberry Pi 3(树莓派3)。树莓派是世界上最流行的单板计算机,3 Model B是最新的版本,我们后面就使用它给大家介绍Android Things。
点此阅读全文
【iOS 开发】 iOS开发tips-UITableView、UICollectionView行高/尺寸自适应
作者:崔江涛
我们都知道UITableView从iOS 8开始实现行高的自适应相对比较简单,首先必须设置estimatedRowHeight给出预估高度,设置rowHeight为UITableViewAutomaticDimension(注意:如果不修改rowHeight默认就是UITableViewAutomaticDimension),对于这两个参数除了直接修改tableview对应的属性之外仍然支持使用对应的代理方法设置。最后只要在UITableViewCell中设置contentView的约束即可。
点此阅读全文
【机器学习】 Tensorflow:tSNE数据非线性降维
作者:artzers
深度学习巨头之一的Hinton大神在数据降维领域有一篇经典论文Visualizing Data using t-SNE。该方法是流形(非线性)数据降维的经典,从发表至今鲜有新的降维方法能全面超越。该方法相比PCA等线性方法能有效将数据投影到低维空间并保持严格的分割界面;缺点是计算复杂度大,一般推荐先线性降维然后再用tSNE降维。Python sklearn有相应的实现。我现在用Tensorflow实现这个算法。
点此阅读全文
【架构】 回顾Bob大叔的简洁架构
作者:半吊子全栈工匠
Robert Martin 就是我们常说的Bob大叔,是码界的骨灰级人物了,在4年前提出了所谓的简洁架构,值得回顾反思一下,看看是否可以借鉴到微服务中呢?
点此阅读全文
【好书推荐】 Mesos和Docker的集成
作者:博文视点
众所周知,Mesos全面支持Docker。但是这意味着什么呢?在命令行里运行docker run…就可以使用Docker了。还需要做什么?让我们一起研究下Mesos的高级特性——和Docker的集成。
本文选自《用Mesos框架构建分布式应用》。
CSDN日报20170227——《什么样的离开会让老板念念不忘》
CSDN日报20170226——《你离心想事成只差一个计划》
CSDN日报20170224——《程序员该用哪种姿势来理财》
CSDN日报20170223——《作为开发者,你都听产品经理的,做的累不累?》
CSDN日报20170220——《从安卓调整到服务端后的思考》
CSDN日报20170218——《你真的看懂无领导小组面试了吗?》
CSDN日报20170217——《辞职信:写给我的“藤野先生”》
关注专栏【CSDN 日报】,获取最新内容。
以上是关于一次fork引发的惨案!的主要内容,如果未能解决你的问题,请参考以下文章
CSDN日报20170301——《一次dns缓存引发的惨案》