Java虚拟机编译《二》控制指令 自己动手实现java虚拟机
yuyutoo 2024-10-23 16:40 2 浏览 0 评论
while循环
void whileInt() {
int i = 0;
while (i < 100) {
i++;
}
}
编译成
Method void whileInt()
0 iconst_0
1 istore_1
2 goto 8
5 iinc 1 1
8 iload_1
9 bipush 100
11 if_icmplt 5
14 return
请注意,在Java虚拟机编写的循环代码中,while语句的判断条件(通过if_icmplt指令实现)是放在循环体的最后面的(早先提到的spin示例也是这样)。将判断条件放在循环体底部,意味着在进入循环的第一次迭代前,必须通过一个goto指令跳转到判断条件那里。如果判断条件不满足,导致循环体一次都未执行,那么这个额外的指令就会被浪费。然而,while循环通常用于预期会多次执行循环体的场景。在接下来的迭代中,将判断条件放在循环底部可以在每次循环时节省一条Java虚拟机指令:如果判断条件放在了循环的开始处,每次循环体执行完毕后都需要一条额外的goto指令来跳转回循环开始处。
涉及其他数据类型的控制结构以类似的方式编译,但必须使用可用于这些数据类型的指令。这会导致代码效率较低,因为需要更多 Java 虚拟机指令,例如
void whileDouble() {
double i = 0.0;
while (i < 100.1) {
i++;
}
}
编译成
Method void whileDouble()
0 dconst_0 // 加载double类型常数0.0
1 dstore_1 // 将double类型数值存储到局部变量1
2 goto 9 // 跳转到指令9
5 dload_1 // 从局部变量1中加载一个double类型值
6 dconst_1 // 加载double类型常数1.0
7 dadd // 执行double类型的加法
8 dstore_1 // 将结果存储回局部变量1
9 dload_1 // 再次从局部变量1中加载double类型值
10 ldc2_w #4 // 加载double类型常数100.1
13 dcmpg // 执行比较操作...
14 iflt 5 // 如果小于0,则跳转到指令5
17 return // 方法返回
在Java的浮点类型中,每一种类型都有两种特定的比较指令,这用于比较两个值的大小。对于float类型,这两个指令是fcmpl和fcmpg;对于double类型,这两个指令是dcmpl和dcmpg。这些指令之间的主要区别在于它们处理非数字(NaN)的方式。
对于NaN,我们认为它是无序的,也就是说,它既不大于、不小于也不等于任何值,包括其自身。因此,如果操作数之一是NaN,那么所有的浮点数比较操作将会失败。
然而为了保证程序的稳定运行,编译器会选择适当的比较指令,确保无论比较在非NaN值上失败还是遇到NaN值时,程序都能得到相同的结果。因此,实际的比较操作结果并不会随着值的差异(NaN或者非NaN)而改变。例如,对于一个比较操作,如果在非NaN值上失败,那么在遇到NaN值时,预期也是失败的。
int lessThan100(double d) {
if (d < 100.0) {
return 1;
} else {
return -1;
}
}
编译成
Method int lessThan100(double)
0 dload_1 // 从局部变量1中加载double类型值
1 ldc2_w #4 // 加载double类型常数100.0
4 dcmpg // 如果输入值是NaN或大于100.0,将1压入栈中;如果输入值等于100.0,将0压入栈中
5 ifge 10 // 如果栈顶数值不小于0(即0或1),跳转到指令10
8 iconst_1 // 将int类型的常量1压入栈中
9 ireturn // 返回栈顶int类型值
10 iconst_m1 // 将int类型的-1压入栈中
11 ireturn // 返回栈顶int类型值
如果d不是NaN并且小于100.0,dcmpg指令会将一个int型-1压入操作数栈中,而ifge指令不会进行跳转。如果d大于100.0或者是NaN,dcmpg指令会将一个int型1压入操作数栈中,此时ifge指令会执行跳转。如果d等于100.0,dcmpg指令会将一个int型0压入操作数栈中,此时ifge指令也会进行跳转。
如果比较相反,dcmpl 指令也能达到同样的效果:
int greaterThan100(double d) {
if (d > 100.0) {
return 1;
} else {
return -1;
}
}
编译成
Method int greaterThan100(double)
0 dload_1 // 从局部变量1中加载double类型值
1 ldc2_w #4 // 加载double类型常数100.0
4 dcmpl // 如果输入值是NaN或小于100.0,将-1压入栈中;如果输入值等于100.0,将0压入栈中
5 ifle 10 // 如果栈顶数值不大于0(即0或-1),跳转到指令10
8 iconst_1 // 将int类型的常量1压入栈中
9 ireturn // 返回栈顶int类型值
10 iconst_m1 // 将int类型的-1压入栈中
11 ireturn // 返回栈顶int类型值
cmp指令总结:Java字节码中dcmpl指令的行为以及其对后续条件分支指令的影响,特别是在处理NaN(不是一个数字)值时的特别之处。
首先,当两个double型值进行比较时,dcmpl指令被用于验证这两个值的大小关系。如果第一个操作数小于第二个,会将-1压入操作数栈;如果两者相等,会压入0;如果第一个操作数大于第二个,会压入1。
然而,比较涉及NaN时,情况有所不同。NaN是指不是一个数字的值,通常作为无效操作的结果出现,比如0除以0。在浮点数比较中,任何涉及NaN的操作应当被视为未定义或失败。dcmpl指令在这种情况下会将-1压入操作数栈,表示比较失败或第一个操作数“小于”第二个操作数。这是为了保证程序能在遇到NaN时仍然能有确定的行为模式。
ifle是一个条件分支指令,它会检查操作数栈顶的值是否小于或等于0,如果是的话,就会跳转到指定的指令地址。因此,如果dcmpl指令将-1压入操作数栈(无论是因为实际的比较结果,还是因为操作数中有NaN),ifle指令就会触发跳转。
如果没有dcmpl和dcmpg这两个指令,程序员就必须自行编写额外的代码来手动检测参数是否为NaN,并据此决定是否跳转,这无疑增加了编程复杂度。简单来说,dcmpl指令简化了在面对NaN值时的比较逻辑,使得处理特殊情况变得更加直接和高效。
相关推荐
- 史上最全的浏览器兼容性问题和解决方案
-
微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●...
-
- 平面设计基础知识_平面设计基础知识实验收获与总结
-
CSS构造颜色,背景与图像1.使用span更好的控制文本中局部区域的文本:文本;2.使用display属性提供区块转变:display:inline(是内联的...
-
2025-02-21 16:01 yuyutoo
- 写作排版简单三步就行-工具篇_作文排版模板
-
和我们工作中日常word排版内部交流不同,这篇教程介绍的写作排版主要是用于“微信公众号、头条号”网络展示。写作展现的是我的思考,排版是让写作在网格上更好地展现。在写作上花费时间是有累积复利优势的,在排...
- 写一个2048的游戏_2048小游戏功能实现
-
1.创建HTML文件1.打开一个文本编辑器,例如Notepad++、SublimeText、VisualStudioCode等。2.将以下HTML代码复制并粘贴到文本编辑器中:html...
- 今天你穿“短袖”了吗?青岛最高23℃!接下来几天气温更刺激……
-
最近的天气暖和得让很多小伙伴们喊“热”!!! 昨天的气温到底升得有多高呢?你家有没有榜上有名?...
- CSS不规则卡片,纯CSS制作优惠券样式,CSS实现锯齿样式
-
之前也有写过CSS优惠券样式《CSS3径向渐变实现优惠券波浪造型》,这次再来温习一遍,并且将更为详细的讲解,从布局到具体样式说明,最后定义CSS变量,自定义主题颜色。布局...
- 你的自我界限够强大吗?_你的自我界限够强大吗英文
-
我的结果:A、该设立新的界限...
- 行内元素与块级元素,以及区别_行内元素和块级元素有什么区别?
-
行内元素与块级元素首先,CSS规范规定,每个元素都有display属性,确定该元素的类型,每个元素都有默认的display值,分别为块级(block)、行内(inline)。块级元素:(以下列举比较常...
-
- 让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华
-
去年的两会期间,习近平总书记在参加人大会议四川代表团审议时,对治蜀兴川提出了明确要求,指明了前行方向,并带来了“祝四川人民的生活越来越安逸”的美好祝福。又是一年...
-
2025-02-21 16:00 yuyutoo
- 今年国家综合性消防救援队伍计划招录消防员15000名
-
记者24日从应急管理部获悉,国家综合性消防救援队伍2023年消防员招录工作已正式启动。今年共计划招录消防员15000名,其中高校应届毕业生5000名、退役士兵5000名、社会青年5000名。本次招录的...
- 一起盘点最新 Chrome v133 的5大主流特性 ?
-
1.CSS的高级attr()方法CSSattr()函数是CSSLevel5中用于检索DOM元素的属性值并将其用于CSS属性值,类似于var()函数替换自定义属性值的方式。...
- 竞走团体世锦赛5月太仓举行 世界冠军杨家玉担任形象大使
-
style="text-align:center;"data-mce-style="text-align:...
- 学物理能做什么?_学物理能做什么 卢昌海
-
作者:曹则贤中国科学院物理研究所原标题:《物理学:ASourceofPowerforMan》在2006年中央电视台《对话》栏目的某期节目中,主持人问过我一个的问题:“学物理的人,如果日后不...
-
- 你不知道的关于这只眯眼兔的6个小秘密
-
在你们忙着给熊本君做表情包的时候,要知道,最先在网络上引起轰动的可是这只脸上只有两条缝的兔子——兔斯基。今年,它更是迎来了自己的10岁生日。①关于德艺双馨“老艺...
-
2025-02-21 16:00 yuyutoo
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)