作者 | 刘欣

本文经授权转自公众号“码农翻身”

JavaScript 的进攻

公元 2014 年,Java 第八代国王终于登上了王位。

第一次早朝,国王坐在高高的宝座上,看着毕恭毕敬的大臣,第一次体会到了皇权的威力。

德高望重的 IO 大臣颤悠悠地走上前来:“启禀陛下,昨日收到战报,有个叫做 Node.js 的番邦又一次向我国进攻,我边防将士死伤惨重。”

“Node.js? 那是什么东西?”  国王心中一乐, 还真有人自不量力,想蚍蜉撼树。 想我 Java 帝国人口之众多,疆域之广阔,踩死你小番邦还不像踩死一只蚂蚁似的。

“那是用 JavaScript 写的一个框架。”  IO 大臣看到国王不知道 Node.js,心里一沉。

“JavaScript?爱卿说笑了,一个在浏览器中运行的东西,怎么可能进攻我 Java 后端。”

“陛下有所不知,这 JavaScript 发展迅猛,不仅占领了前端,还通过 Node.js 向后端,尤其是我国渗透,臣还听说他们用 Electron 开始蚕食桌面开发了!”

“竟有这等事!难道他们想通吃?我们不是有 Tomcat 吗?派 Tomcat 去把 Node.js 给镇压了。”

国王开始怨恨自己的父亲 JDK 7 世和祖父 JDK 6 世没把这个 Node.js 当成一回事,没有把 Node.js 给扼杀在摇篮之中,把这个祸害留给了自己,心里开始发虚。

非阻塞异步 IO

线程大臣走上前来:“陛下,Tomcat 已经率军和 Node.js 恶战了几日,败下阵来, 这 Node.js 有个独门武器,叫做‘非阻塞异步 IO’。”

“非阻塞? 我听说我们的 Tomcat 也能实现非阻塞啊!”  国王有点惊讶。

“不行的,陛下,Tomcat 在处理连接的时候能实现非阻塞,但是在真正处理请求的时候还是需要同步操作,一个请求对应一个线程来处理,不像 Node.js 那样,都是异步操作,只有一个主线程在忙活。”  线程大臣做了一个简明扼要的汇报,不知道国王能否听懂。

注: Node.js的故事请参见《Node.js: 我只需要一个店小二》

“众位爱卿,你们说说该怎么办? 总不能让这小小番邦屡次欺负我堂堂 Java 帝国吧。”

“臣倒是有一计,” 集合大臣说道,“这 Node.js 虽然来势汹汹,但是它也有个致命的缺点,那 JavaScript 是个动态语言,无法进行编译时类型检查,错误只有等到运行时才能暴露出来。用它开发个小项目还可以,一旦项目变大,代码变多,人员变多,那就会变成噩梦了。”

“爱卿说说具体怎么办?”

“我们可以派一些卧底去 Node.js, 到处传播这样的消息,瓦解他们的军心和士气,让他们认为 Node.js 写的系统,很快就会腐化,最终还是要用我堂堂正正的 Java 语言来重写。”

“嗯,此乃心理战也,至少会稳住一些墙头草,准奏,由爱卿来安排。 ” 国王说道,“不过,此法治标不治本,还是得想办法直接把他们打败。”

“陛下真乃一代圣君,” 线程大臣马上开始拍马屁,与此同时,巧妙地把矛头转向老不死的 IO 大臣:“我 Java 帝国在第 4 代国王的时候就出现了非阻塞 IO,这么多年过去了,居然还没发展出类似 Node.js 的系统,实在是不应该啊。”

“老不死”的 IO 大臣是何等精明:“陛下明鉴, 我 Java 帝国应用服务器一直以来都是 Tomcat 独大,他们采用了线程池,每个请求一个线程的方式,我也不好干预。”

IO 大臣把责任推得一干二净。

“没错,” 集合大臣为 IO 大臣打抱不平,两肋插刀,“还有一点就是这异步编程,听起来很好,但是写起来可就要命了,那么多的回调,简直就是反人类,臣民们戏称为回调地狱,没人愿意那么写,发展不起来也很正常。”

线程大臣马上接口:“此言差矣,陛下已经教会了臣民们如何使用 Lambda 表达式,并且现在也出现了 RxJava,已经没什么回调地狱了!”

“那是现在,以前可没有!”

“......”

国王看到这几位大臣要打起来,马上施展和稀泥之术:“众位爱卿各有道理,你们且说说,怎么才能打败着来势汹汹的 Node.js 吧。”

没人说话。

国王只好退朝。

京城酒馆

京城的小酒馆向来是一个多方消息的集散地。

一个金发碧眼的小伙子正在“危言耸听”:“听说了没有,Node.js 又赢了几仗,Tomcat 大军死伤惨重,有不少臣民都投奔到那个番邦去了。”

“这异步操作真的有这么厉害?” 有人问道。

小伙子喝了一口酒: “其实不是异步操作更好,而是在高并发的环境异步操作更有效,大家都知道, 一个机器能支持的线程数目是有限的,不可能一直增加。Tomcat 那种一个请求一个线程的方式很快就会遇到瓶颈。”

“你说说,到底有什么好处?”有人刨根问底。

“现在服务器端的操作无非就是操作文件,读写数据库,访问远程服务,这些都是所谓阻塞操作。” 小伙子展开了一张图:

“橙色的都是 IO 操作,绿色的才是真正的线程执行, IO 操作非常耗时,线程大部分时间都浪费在了等待上面!如果能让线程不要等待,去做别的事情,那用少量的线程,甚至一个线程就可以了。”

众人纷纷点头, 这小伙子已经看出了问题的关键,现在的很多系统,都是 IO 密集的, 高并发情况下,如果一个请求一个线程,浪费巨大。

“想我 Java 虚拟机如此强悍,如果能实现异步操作,那还不把 Node.js 秒成渣?!”小伙子狠狠地用手锤了一下桌子。

正在此时,酒馆冲进一队士兵,赶走众人,围住小伙子,领头的喝问到:“大胆刁民,竟然到处宣扬异步思想,给我带走!”

士兵恶狠狠地把他五花大绑,推出门去, 留下一堆人在那里议论纷纷。

IO 王府

“我让你们把他请来,怎么绑来了?快松绑!” IO 大臣呵斥完下属,转头亲切地问道:“叫什么名字啊?”

“小人蒂姆, Tomcat 府上的幕僚。 ” 蒂姆一边说一边揉肩膀。

“Tomcat 府上的人......”  IO 大臣捻着胡须若有所思。

“是的,大人,我还见过您呢,您上次半夜去 Tomcat 府上密谈......”

“住口! ” IO 大臣赶紧转换话题, “我的下属发现你到处宣扬异步思想,究竟要干什么? ”

“小人发明了一个系统,叫做 Node.x。 ”

“为什么不献于 Tomcat 将军?”

“唉,小人进言多次,可是将军不听啊!”

“你说说看,这是个什么东西?  是要模仿 Node.js 吗?”  IO 大臣问道。

之前蒂姆给 Tomcat 将军讲述过 Node.js, 他理都不理,经常是一甩袖子就走, 自己是空有一身本领却无人赏识, 难道这 IO 大臣能帮自己一把? 想到此处,蒂姆精神大振。

“确实受到了它的启发, 但是我的 Node.x 在架构和一些关键的抽象上和 Node.js 有很大不同。” 蒂姆不好意思地笑了笑,“先说说相同的部分,既然都是异步操作,那肯定是通过事件驱动的,所以都有一个事件循环。”

IO 大臣之前和 Swing 大臣聊过, 知道事件循环是怎么回事,这是一个相当古老的概念了。

无非就是有个线程在检测一个队列,如果队列中有事件,就拿出来处理。

“只不过我这里有所不同,可以创建多个事件循环出来,比如每一个 CPU 核心有一个,这样可以充分利用 CPU 的多核性能。”  蒂姆得意地说道。

(4 个 CPU Core, 4 个事件循环)

IO 大臣点头表示赞许, 他听说 Node.js 好像只有一个主线程,没法直接利用多核的能力。想利用多核的话还得开多个进程才行。

异步操作

“你图中的那个 Hanlder 就是具体的业务代码所在地吧? 具体长什么样子啊,让我看看!”  IO 大臣问道。

蒂姆赶紧呈上代码,这是简单的 Hello World。

import io.vertx.core.AbstractVerticle;public class Server extends AbstractVerticle {  public void start() {    vertx.createHttpServer().requestHandler(req -> {      req.response()        .putHeader("content-type", "text/plain")        .end("Hello Word!");    }).listen(8080);  }}

这段代码生成了一个简单的 HTTP 服务器, 在 8080 端口监听, 每当有请求来的时候,都返回一个字符串“Hello World!”。

IO 大臣一看,大为吃惊:“你这代码不需要外部容器,自己就搞了一个 HTTP 服务器啊?”

“是的,这样我们就完全不用 Tomcat 了。我把这种类起了一个名称,叫做 Verticle, 部署以后,这个 Verticle 就可以和一个事件循环关联了。每次有 HTTP 请求过来,Node.x 会封装成事件,然后分派给它处理了。”

真是个二愣子, IO 大臣心想, 怪不得 Tomcat 对你不待见,你这个东西出来,他的位置不保啊!

IO 大臣问道:“那对于数据库查询,你这个 Handler,哦不,Verticle 该怎么写? 查询数据库这么慢,岂不是把事件循环都阻塞了?什么事情都做不了了?”

“大人您忘了,我们这里操作必须都是异步的,查询数据库也不例外。”

蒂姆说着展示了一段代码, 通过异步的方式来查询数据库。

public class DatabaseVerticle extends AbstractVerticle{    ......    dbClient.getConnection(ar -> {        if (ar.succeeded()) {            SQLConnection connection = ar.result();            connection.query("select .. from...", res -> {                       if (res.succeeded()) {                   ......                 } else {                   ......                 }          });        } else {           ......        }    });}

IO 大臣感慨道:“唉,老了,真是不中用了,连异步都忘了。对了,这些个 Verticle 看起来都是独立的,是被不同的线程调用的,他们之间怎么进行交互啊?难道也通过共享内存的方式?”

“大人真是厉害,一下子就问到了核心问题,不能让他们共享内存,那样就需要加锁了,我这里引入了 Event Bus 的方法,让他们之间通过消息传递。”

“嗯,不错,实现了低耦合。”

“不仅如此,这些 Verticle 还可以部署到不同的 JVM 中,通过 Event Bus 实现真正的分布式通信。” 蒂姆又抛出一个重磅炸弹。

“如此甚好!”  IO 大臣爱才之心骤起, “你愿不愿意到老夫府上做幕僚啊?”

“小人愿意追随大人!”

“好!明日早朝,你随我入宫,面见圣上,老夫保你一世荣华富贵。 ”

为什么是 Vert.x?

第二日早朝,IO 大臣迫不及待地给国王报喜:“陛下,我 Java 帝国也可以采用非阻塞异步编程了!击败 Node.js 之日可待。”

IO 大臣讲述了昨晚的情况, 细数了 Node.x 的种种好处。

Tomcat 将军脸上极为难看, 赶紧阻止:“陛下不可,我 Java 帝国采用同步处理已经很久了,臣民们已经习惯了,现在改成异步,怕激起民变。”

“爱卿不要低估臣民采用新技术的能力嘛, 宣蒂姆进殿,呈上代码。”

蒂姆都不敢看 Tomcat, 从怀里掏出一张纸,双手奉上。

vertx.createHttpServer()  .requestHandler(function (req) {    req.response()      .putHeader("content-type", "text/plain")      .end("Hello World");}).listen(8080);

国王盯着看了半天:“嗯?不对啊,你这不是 Java 代码吧?”

Tomcat 拿过国王递过来的代码,扫了一眼:““大胆! 你竟然敢在朝堂之上公然宣传 JavaScript,来人,拿下!”

“陛下息怒,这是小人制定的一个策略,我的 Node.x 支持很多语言编程, 除了 Java 之外,还有 JavaScript、Ruby、Scala、Kotlin 等等。”

“哦? 是吗? 这还能把番邦的人给吸引过来呢!你说呢,Tomcat 将军?” 国王说道。

Tomcat 有些不自在,想找回场子:“嗯嗯,有一定道理,不过这个 Node.x 这个名字不好,拾人牙慧,让人看低我堂堂 Java 帝国。”

“Node 是节点的意思,朕把他改成 Vertex 如何?也是节点的意思。”

“ 陛下圣明,可否叫做 Vert.x ? ” IO 大臣提议。

“好,准奏,即日起,命你和蒂姆训练臣民使用 Vert.x,一个月后向 Node.js 开战!”  国王已经忍 Node.js 很久了。

不,不能让 IO 大臣的 Vert.x 一家独大!

国王突然想到了亲爹留下来的祖训, 帝王之术是一定要平衡朝局。

“吩咐下去,今晚朕要和 Spring 将军,嗯,还有线程大臣,共进晚餐,朕有些事情要和他们好好谈谈......”

*本故事纯属虚构,如有雷同,纯属巧合。

声明:作者独立观点,不代表 CSDN 立场。


 热 文 推 荐 

☞ 2018 中国互联网哈哈榜之「十大 CP」

☞ 极客头条

☞ “离开 360 时,它只给了我一块钱”

☞ 与 30 家公司过招,得到了这章面试心法

☞ 断舍离:我彻底戒掉苹果、微软、Google、Facebook 和亚马逊之后?

☞ 微信说不!有何不可?

☞ 11.73 亿人看央视春晚;苹果逼用户买充电器;特斯拉 Model 3 降价七千 | 极客头条

☞ 移动互联网“大开放”之死

☞ @程序员,沟通这项核心技能你掌握了多少?

年薪 10 万的程序员,如何积累人生的第一个 100 万?

print_r('点个好看吧!');
var_dump('点个好看吧!');
NSLog(@"点个好看吧!");
System.out.println("点个好看吧!");
console.log("点个好看吧!");
print("点个好看吧!");
printf("点个好看吧!");
cout << "点个好看吧!" << endl;
Console.WriteLine("点个好看吧!");
fmt.Println("点个好看吧!");
Response.Write("点个好看吧!");
alert("点个好看吧!")
echo "点个好看吧!"

点击“阅读原文”,打开 CSDN App 阅读更贴心!

喜欢就点击“好看”吧!

Java 能抵挡住 JavaScript 的进攻吗?相关推荐

  1. Java能抵挡住JavaScript的进攻吗?

    JavaScript的进攻 公元2014年,Java 第八代国王终于登上了王位. 第一次早朝,国王坐在高高的宝座上,看着毕恭毕敬的大臣,第一次体会到了皇权的威力. 德高望重的IO大臣颤悠悠地走上前来: ...

  2. Java能抵挡住JavaScript的进攻吗?【转载】

    JavaScript的进攻 公元2014年,Java 第八代国王终于登上了王位. 第一次早朝,国王坐在高高的宝座上,看着毕恭毕敬的大臣,第一次体会到了皇权的威力. 德高望重的IO大臣颤悠悠地走上前来: ...

  3. Java程序猿的JavaScript学习笔记(12——jQuery-扩展选择器)

    计划按例如以下顺序完毕这篇笔记: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScript ...

  4. 2018最具就业前景的7大编程语言,Java、Python和JavaScript?

    2018 年即将到来,Coding Dojo(编码道场)近期发布了 2018 最具就业前景的 7 大编程语言.该公司分析了来自 Indeed 的 25 门编程语言.栈和框架的数据,以找出雇主最需求的七 ...

  5. Java程序猿的JavaScript学习笔记(汇总文件夹)

    最终完结了,历时半个月. 内容包含: JavaScript面向对象特性分析,JavaScript高手必经之路. jQuery源代码级解析. jQuery EasyUI源代码级解析. Java程序猿的J ...

  6. java rhino import_java调用javascript :js引擎rhino

    前段时间,在浏览javaeye论坛,看见有人征集如何在java中运行数学表达式. 结果方案五花八门: 1.jakarta commons JEXL. 2.Beanshell 3.Java Math E ...

  7. Java程序猿的JavaScript学习笔记(10—— jQuery-在“类”层面扩展)

    计划按例如以下顺序完毕这篇笔记: Java程序猿的JavaScript学习笔记(1--理念) Java程序猿的JavaScript学习笔记(2--属性复制和继承) Java程序猿的JavaScript ...

  8. tea java 代码,TeaVM编译器如何将Java应用程序转换为Javascript,以及如何使用MicroK8管理嵌入...

    阅读这篇文章你会学到: 1.TeaVM如何将Java应用程序转换为Javascript. 2.如何模仿Nvidia的Jetson. 3.使用MicroK8管理嵌入式应用程序中的容器. 首先我先来介绍一 ...

  9. 2018年最具就业前景的7大编程语言:Java、Python、JavaScript、C++、C#、PHP、Perl ......

    软件开发是一个动态的领域.新的编程语言.框架和技术陆续出现,流行,然后消失.开发者需要不断学习新技能,掌握企业发展所需要的新技术才能不被淘汰. 近日,Coding Dojo 分析了来自Indeed.c ...

最新文章

  1. linux 查看es安装目录,Linux安装Elasticsearch
  2. 2.3.10 哲学家进餐问题
  3. 思科安全——企业安全棋局的“宇宙流”
  4. php连接mysql并读取数据
  5. 端到端加密优缺点_基于Filecoin的去中心化文件保存和加密分享平台
  6. 详解Python切片技术
  7. 中国3亿中年男人都在花钱买什么?数据分析后告诉你答案,很现实
  8. asp access服装商城计算机设计网站作品
  9. java 弹幕游戏_java弹幕小游戏1.0版本
  10. PHP判断浏览器类型及版本
  11. STM32单片机最小系统
  12. 微信H5开发,怎样禁止手机横屏
  13. Web.config配置文件详解(转载)
  14. Spring ClassPathResource详解
  15. 模糊测试中的动态符号执行
  16. linux查找文件或文件夹
  17. 30款android开发高效必备工具(附下载地址)
  18. KVO实现机制 如何自己动手实现 KVO
  19. linux中从内存解析xml格式的库,Linux下使用libxml库解析xml文件
  20. python爬虫实战三:近十年中国电影票房数据爬取与分析

热门文章

  1. 矩池云上nvidia opencl安装及测试教程
  2. python捕捉warning_python – 如何格式化logging.captureWarnings捕获的警告?
  3. 显卡显存故障检测工具_为RTX30系显卡做准备,骨伽GEX750金牌全模组电源装机体验...
  4. 命令窗口ping oracle,Oracle中tnsping命令解析
  5. c++分治法求最大最小值实现_你所不了解的分治算法
  6. Flutter进阶—质感设计之表单输入
  7. 等级VIN行业调研报告 - 市场现状分析与发展前景预测(2021-2027年)
  8. 免校准的电量计量芯片_安科瑞专场5G基站用电改造直流电能计量表AMC16DETT
  9. 拒绝假货!LVMH与普拉达、卡地亚联手推出区块链平台AURA
  10. 经典面试题:有序矩阵的快速查找