那天,电脑里的两个家伙吵起来了···

编程技术宇宙

共 1798字,需浏览 4分钟

 ·

2020-08-14 20:10


某天周末,CPU和JVM聊了起来···




小兄弟,最近听说你风头很猛啊

大哥见笑了,岂敢岂敢




岂敢?我看你敢的很嘛,自己搞了一套虚拟机,都快把我架空了

大哥你看你说的,我就是一Java虚拟机,最终还得把代码指令发给您来执行。根本离不开你,怎么会架空你呢




我倒是很好奇,既然最终还是得靠我来执行指令,那你存在的意义又是什么呢?

这您就有所不知了。我作为中间商,为程序们提供了最优质便捷的服务,要是让他们直接跟你打交道,那还得了解你的指令集,你的I/O方式、你的内存管理方式,可麻烦了。





别蒙我了,你说的这些,操作系统都封装好了,程序们直接调用就好了,哪有那么麻烦?

。。。。。。


你说的不错,不过有一个致命的问题,就是操作系统也帮不上忙。






什么问题?

跨平台移植





你是说在移植到不同的操作系统上吗?

不仅仅是操作系统,还可能移植到别的架构CPU上。这样原来的程序就没法执行了。






提问题当然容易,那你有什么好的解决办法?

我的办法就是不要让程序代码编译成你的指令集的指令,而是生成一个我定义的中间码。运行的时候再由我来负责把它们翻译成当前平台下对应的CPU指令,交给你们CPU来执行。





好家伙,还说不是把我架空,这样一来,你这个中间商好赚差价啊。

不过这对您也没什么损失嘛,反正最后还是躲不开您这一层。





我听你这意思,那如果其他语言能编译成你的中间码,你也能执行了?

那可不,除了Java,咱还能执行Groovy、Scala,对了,Python咱也能执行。这不跟您一样嘛,不管是什么高级语言,最终编译成你的指令集,你就能执行了,这不一个道理嘛





有点6啊,你这个中间码是什么样的?有这么厉害,跟我的指令集有什么不同呢?

跟您的指令集很像,不同的是我的“指令”不像您的指令有长有短,我的统一都是一个字节,所以也叫字节码!





都是一个字节?别蒙我了,光内存地址都不止一个字节。

真的,我基本上都是零地址指令,要用的数据都在栈顶,直接操作就是了





都在栈顶?不用寄存器怎么进行数据操作呢?

谁说一定要有寄存器?我就不用寄存器,您是基于寄存器架构的,而我是基于栈式架构的。运算的时候直接操作栈顶的数据就行了






基于栈?你是说你的指令用到的数据都放在栈里是吗?这栈是在内存中,这样的话数据不停进进出出就要频繁访问内存,肯定很慢吧?

那肯定是比您慢了,不过我也有优化办法。





什么办法,说来听听

Top-Of-Stack-CAching,简称TOSCA,就是栈顶缓存。在栈顶的数据虽然名义上是在栈顶,但实际我在实现的时候,偷偷的放在了你的寄存器里,不仅可以减少访问内存的次数,你到时候执行的时候访问起来也快得多。





好家伙,你这小子当面一套,背后一套啊。





你开始说到了“翻译”,你是怎么翻译执行这些字节码的呢?

我们为每个字节码都准备了对应的汇编指令,来完成这个字节码定义的行为。然后把所有字节码的汇编指令入口整合到一张表里来,我来统一派遣分发就可以了。这些汇编指令到时候就直接交给你CPU来执行就好啦。




我知道了,你们的跨平台关键就是在不同平台生成对应的的汇编指令,是吧!





不过这听起来有点像是解释执行啊,跟那些脚本语言解释执行差不多嘛

你说的那些脚本语言底层是用高级语言在“模拟”执行,而我是直接把字节码翻译成了汇编指令,比他们快多了。





再快,那也是解释执行

好吧,看来得亮出绝活了!





啥?

JIT,just-in-time compilation,即时编译技术。





这是啥,看上去很厉害的样子

我在派遣分发的过程中,发现有频繁执行的函数或者代码块,我就会启动即时编译机制,将这部分代码直接编译成你CPU的指令,后面就不用再走字节码派遣表执行了





频繁执行的函数好理解,频繁执行的代码块是什么意思?

比如函数中某个循环体,循环执行很多次,虽然可能这个函数调用次数不多,但循环执行要是很多,也会触发我的即时编译机制





你是怎么判定是否是频繁呢,不会是凭直觉吧

当然不是,频不频繁,我心里是有“数”的。这个数就是计数器,我会统计函数的调用次数和循环体的执行次数。




好小子,有两把刷子啊,没少花心思

雕虫小技,不敢班门弄斧








呀!别聊了,程序员开机了,要去干活了

今天不周末吗,他咋还来




浏览 9
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报
评论
图片
表情
推荐
点赞
评论
收藏
分享

手机扫一扫分享

分享
举报