多线程:第一章:我(线程)这一辈子

一个故事帮你理解线程和线程池

我是一个线程, 我一出生就被编了个号: 0×3704,  而后被领到一个昏暗的屋子里,  这里我发现了不少和我如出一辙的同伴。html

我身边的同伴0×6900 待的时间比较长, 他带着沧桑的口气对我说:算法

咱们线程的宿命就是处理包裹。 把包裹处理完之后还得立刻回到这里,不然可能永远回不来了。数据库

我一脸懵懂,包裹,什么包裹?缓存

”不要着急,立刻你就会明白了, 咱们这里是不养闲人的。“网络

果真,没多久,屋子的门开了, 一个面貌凶恶的家伙吼道:多线程

"0×3704 ,出来!"分布式

我一出来就被塞了一个沉甸甸的包裹,上面还有附带着一个写满了操做步骤的纸。网站

"快去,把这个包裹处理了。"操作系统

"去哪儿处理"线程

"跟着指示走, 先到就绪车间"

果真,地上有指示箭头,跟着它来到了一间明亮的大屋子,这里已经有很多线程了, 你们都很紧张,好像时刻准备着往前冲。

我刚一进来,就听见广播说:“0×3704, 进入车间”

我赶忙往前走, 身后不少人议论说:

”他太幸运了, 刚进入就绪状态就能运行“

”是否是有关系?“

”不是,你看人家的优先级多高啊, 唉“

前边就是车间, 这里简直是太美了, 怪不得老线程老是唠叨着说:要是能一直待在这里就行了。

这里空间大,视野好,空气清新,鸟语花香,还有不少历来没见过的人,像服务员同样等着为我服务。

他们也都有编号, 更重要的是每一个人还有个标签,上面写着:硬盘,数据库,内存,网卡...

我如今理解不了,看看操做步骤吧:

第一步:从包裹中取出参数  

打开包裹, 里边有个HttpRequest  对象, 能够取到 userName, password两个参数

第二步:执行登陆操做

奥,原来是有人要登陆啊,我把userName/password 交给 数据库服务员,他拿着数据, 慢腾腾的走了。

 

他怎么这么慢? 不过我是否是正好能够在车间里多待一下子? 反正也无法执行第三步。

就在这时,车间里的广播响了: 

“0×3704,  我是CPU , 记住你正在执行的步骤, 立刻带包裹离开”

我慢腾腾的开始收拾

”快点, 别的线程立刻就要进来了“

离开这个车间, 又来到一个大屋子,这里不少线程慢腾腾的在喝茶,打牌。

”哥们,大家没事干了?“

”你新来的把, 你不知道我在等数据库服务员给我数据啊! ,听说他们比咱们慢好几十万倍, 在这里好好歇吧“

”啊? 这么慢?  我这里有人在登陆系统, 能等这么长时间吗”

”放心,你没据说过人间一天, CPU一年吗, 咱们这里是用纳秒,毫秒计时的, 人间等待一秒,至关于咱们好几天呢, 来的及“

干脆睡一会吧 , 不知道过了多久 ,大喇叭又开始广播了:

“0×3704, 你的数据来了,快去执行”

我转身就往CPU车间跑,发现这里的们只出不进!

后面传来阵阵哄笑声:

”果真是新人, 不知道还得去就绪车间等“

因而赶忙到就绪车间, 此次没有那么好运了, 等了很久才被再次叫进CPU车间。

在等待的时候, 我听见有人小声议论:

”据说了吗,最近有个线程被kill掉了“

”为啥啊?“

”这家伙赖在CPU车间不走, 把CPU利用率一直搞成100%,后来就被kill掉了“

”Kill掉之后弄哪儿去了“

”可能被垃圾回收了吧“

我内心打了个寒噤 ,  赶忙接着处理, 收下的动做块多了,第二步登陆成功了

第三步:构建登陆成功后的主页

这一步有点费时间, 由于有不少HTML须要处理, 不知道代码谁写的,处理起来很烦人。

我正在紧张的制做html呢, CPU有开始叫了:

“0×3704,  我是CPU , 记住你正在执行的步骤, 立刻带包裹离开”

”为啥啊“

”每一个线程只能在CPU上运行一段时间,到了时间就得让别人用了, 你去就绪车间待着, 等着叫你吧“

就这样, 我一直在就绪-运行 这两个状态,不知道轮转了多少次, 终于安装步骤清单把工做作完了。

最后顺利的把包含html的包裹发了回去。

至于登陆之后干什么事儿 , 我就无论了。

立刻就要回到我那昏暗的房间了, 真有点舍不得这里。

不过相对于有些线程, 我仍是幸运的, 他们运行完之后就完全的销毁了,而我还活着 !

回到了小黑屋, 老线程0×6900 问:

”怎么样?第一天有什么感受?“

”咱们的世界规则很复杂 , 首先你不知道何时会被挑中执行;  第二 ,在执行的过程当中随时可能被打断,让出CPU车间;

第三,一旦出现硬盘,数据库这样耗时的操做也得让出CPU,去等待;  第四,就是数据来了,你也不必定立刻执行,还得等着CPU挑选“

”小伙子理解的不错啊“

”我不明白为何不少线程都执行完就死了, 为何我们还活着?“

”你还不知道, 长生不总是咱们的特权, 咱们这里有个正式的名称,叫作 线程池!“

平淡的日子就这么一每天过去, 做为一个线程, 我天天的生活都是取包裹,处理包裹,而后回到咱们昏暗的家:线程池。

有一天我回来的时候, 听到有个兄弟说, 今天要好好休息下,明天就是最疯狂的一天。

我看了一眼日历,明天是 11月11号 。

果真,零点刚过,不知道那些人类怎么了, 疯狂的投递包裹, 为了应付蜂拥而至的海量包裹, 线程池里没有一我的能闲下来,所有出去处理包裹,CPU车间利用率超高,硬盘在嗡嗡转, 网卡疯狂的闪,  即使如此, 仍是处理不完,堆积如山。

咱们也没有办法,实在是太多太多了, 这些包裹中大部分都是浏览页面,下订单,买,买,买。

不知道过了多久, 包裹山终于慢慢的消失了。

终于可以喘口气, 我想我永远都不会忘记这一天。

经过这个事件,我明白了我所处的世界:这是一个电子商务的网站!

我天天的工做就是处理用户的登陆,浏览, 购物车,下单,付款。

我问线程池的元老0×6900 : ” 咱们要工做到何时?”

” 要一直等到系统重启的那一刻”, 0×6900 说

” 那你经历过系统重启吗?”

” 怎么可能? , 系统重启就是咱们的死亡时刻, 也就是世界末日,一旦重启, 整个线程池所有销毁,时间和空间所有消失,一切从头再来”

” 那何时会重启?”

” 这就很差说了,好好享受眼前的生活吧…..”

其实生活丰富多彩, 我最喜欢的包裹是上传图片,因为网络慢,因此能在就绪车间, CPU车间待很长很长时间,能够认识不少好玩的线程。

好比说上次认识了memecached 线程,他给我说经过他缓存了不少的用户数据, 仍是分布式的! 不少机器上都有!

我说怪不得后来的登陆操做快了那么多, 原来是再也不从数据库取数据了你那里就有啊, 哎对了你是分布式的你去过别的机器没有?

他说怎么可能我每次也只能经过网络往那个机器发送一个GET, PUT命令才存取数据而已, 别的一律不知。

再好比说上次在等待的时候遇到了数据库链接的线程, 我才知道它他那里也是一个链接池, 和咱们线程池几乎如出一辙。

他说有些包裹太变态了,居然查看一年的订单数据, 简直把我累死了。

我说拉倒吧你, 你那是纯数据, 你把数据传给我之后,我还得组装成HTML,  工做量不知道比你大多少倍。

他说必定你要和memecached搞好关系,直接从他那儿拿数据,尽可能少直接调用数据库, 咱们JDBC connection也能活的轻松点。

我说好啊好啊, 关键是你得提早把数据搞到缓存啊, 要否则我先问一遍缓存, 没有数据, 我这不还得找你吗?

生活就是这样, 若是你本身不找点乐子,还有什么意思?

有一天我遇到一个可怕的事情, 差一点死在外边,回不了线程池了……

其实此次遇险我应该可以预想到才对, 太大意了。

前几天我处理过一些从http 发来的存款和取款的包裹, 老线程0×6900 特地嘱咐我:

“处理这些包裹的时候要特别当心, 你得必定要先得到一把锁, 在对帐户存款或者取款的时候必定要把帐户给锁住, 要否则别的线程就会在你等待的时候趁虚而入,搞破坏, 我年轻那会儿很毛糙,就捅了篓子”

为了“恐吓”我, 好心的0×6900还给了我两个表格:

一、没有加锁的状况

 

二、加锁的状况

 

我看的胆颤心惊, 原来不加锁会带来这么严重的事故。

今后之后看到存款,取款的包裹就倍加当心, 还好,没有出过事故。

今天我收到的一个包裹是转帐, 从某著名演员的帐号给某著名导演赚钱, 具体是谁我就不透漏了, 数额可真是不小

我按照老线程的吩咐, 确定要加锁啊, 先对著名演员帐号加锁, 在对著名导演帐号加锁。

可我万万没想到的是, 还有一个线程,对,就是0×7954,  居然同时在从这个导演到往这个演员转帐。

因而乎,就出现了这么个状况:

刚开始我还不知道什么状况, 一直坐在等待车间傻等, 但是等的时间太长了, 长达几十秒 ! 我可历来没有经历过这样的事件。

这时候我就看到了线程0×7954 , 他清闲的坐在那里喝咖啡, 我和他聊了起来:

“哥们, 我看你已经喝了8杯咖啡了, 怎么还不去干活?”

“你不喝了9杯茶了吗?” 0×7954 回敬到。

“我在等一个锁, 不知道哪一个孙子一直不释放”

“我也在等锁啊,我要是知道哪一个孙子不释放锁我非揍死他不可 ” 0×7954 绝不示弱。

我偷偷的看了一眼, 这家伙怀里不就抱着我正在等的 某导演的锁嘛?

很明显, 0×7954 也发现了我正抱着他正在等待的锁。

很快咱们两个就吵了起来, 各执己见:

“把你的锁先给我, 让我先作完”

“不行, 历来都是作完工做才释放锁, 如今绝对不能给你”

从争吵到打起来, 就那么几秒钟的事儿。

更重要的是, 咱们俩不只仅持有这个著名导演和演员的锁, 还有不少其余的锁, 致使等待的线程愈来愈多, 围观的人们把屋子都挤满了。

最后事情真的闹大了, 我历来没见过终极大boss “操做系统” 也来了。

大Boss毕竟是见多识广, 他看了一眼, 哼了一声 , 很不屑的说:

“又出现死锁了”

“大家俩要Kill掉一个, 来吧, 过来抽签 ”

这一会儿把我给吓尿了, 这么严重啊!

我战战兢兢的抽了签,打开一看, 是个”活”字。

唉,小命终于保住了。

可怜的0×7954 被迫交出了全部的资源之后, 很不幸的被kill掉, 消失了。

我拿到了导演的锁, 能够开始干活了。

大Boss操做系统如一阵风似的消失了, 身后只传来他的声音:

记住, 咱们这里导演>演员,  不管认识状况都要先得到导演的锁

因为不只仅是只有导演和演员, 还有不少其余人, Boss留下了一个表格,  里边是个算法, 用来计算资源的大小, 计算出来之后,永远按照从大到小的方式来得到锁:

我回到线程池, 你们都知道了个人历险, 围着我问个不停。

凶神恶煞的线程调度员把大Boss的算法贴到了墙上。

 

天天早上, 咱们都得像无节操的房屋中介, 美容美发店的服务员同样, 站在门口,像被耍猴同样大声背诵:

“多个资源加锁要牢记, 必定要按Boss的算法比大小, 而后从最大的开始加锁”

——————————————————–

又过了数日, 我和其余线程们发现了一个奇怪的事情:包裹的处理愈来愈简单

无论任何包裹,无论是登陆, 浏览,存钱….. 处理的步骤都是同样的, 返回一个固定的html页面

有一次我偷偷的看了一眼, 上面写着:

“本系统将于今晚 00:00 至4:00 进行维护升级, 给你带来的不便咱们深感抱歉”

我去告诉了老线程0×6904,  他叹了一口气说:

“唉, 咱们的生命也到头了, 看来立刻就要重启系统, 咱们就要消失了, 再见吧兄弟。”

系统重启的那一刻终于到来了。

我看到屋子里的东西一个个的不见了, 等待车间,就绪车间,甚至CPU车间都慢慢的消失了。

我身边的线程兄弟也愈来愈少, 最后只剩我本身了。

我在空旷的原野上大喊: 还有人吗?

无人应答。

咱们这一代线程池完成了使命。

下一代线程池将很快重生。