java $p_javap -c命令详解

一直在学习Java,碰到了很多问题,碰到了很多关于i++和++i的难题,以及最经典的String str = "abc" 共创建了几个对象的疑难杂症。 知道有一日知道了java的反汇编 命令  javap。现将学习记录做一小结,以供自己以后翻看。如果有错误的地方,请指正

1.javap是什么

where options include:

-c Disassemble the code

-classpath Specify where to find user class files

-extdirs Override location of installed extensions

-help Print this usage message

-J Pass directly to the runtime system

-l Print line number and local variable tables

-public Show only public classes and members

-protected Show protected/public classes and members

-package Show package/protected/public classes

and members (default)

-private Show all classes and members

-s Print internal type signatures

-bootclasspath Override location of class files loaded

by the bootstrap class loader

-verbose Print stack size, number of locals and args for met

hods

If verifying, print reasons for failure

以上为百度百科里对它的描述,只是介绍了javap的一些参数和使用方法,而我们要用的就是这一个:-c Disassemble the code。

明确一个问题:javap是什么?网上有人称之为 反汇编器,可以查看java编译器为我们生成的字节码。通过它,我们可以对照源代码和字节码,从而了解很多编译器内部的工作。

2.初步认识javap

从一个最简单的例子开始:

d89865d4fbf3af304b5060b2e4fa6577.png

这个例子中,我们只是简单的声明了两个int型变量并赋上初值。下面我们看看javap给我们带来了什么:(当然执行javap命令前,你得首先配置好自己的环境,能用javac编译通过了,即:javac TestJavap.java )

13e79958cca70a7178f3ca76ce790925.png

我们只看(方便起见,将注释写到每句后面)

Code:0: iconst_2 //把2放到栈顶

1: istore_1 //把栈顶的值放到局部变量1中,即i中

2: iconst_3 //把3放到栈顶

3: istore_2 //把栈顶的值放到局部变量1中,即j中

4: return

是不是很简单?(当然,估计需要点数据结构的知识) ,那我们就补点java的关于堆栈的知识:

对于 int i = 2;首先它会在栈中创建一个变量为i的引用,然后查找有没有字面值为2的地址,没找到,就开辟一个存放2这个字面值的地址,然后将i指向2的地址。

看了这段话,再比较下上面的注释,是不是完全吻合?

为了验证上面这一说法,我们继续实验:

4e93ff1ba59e36084772a67756f8913c.png

我们将 i 和 j的值都设为2。按照以上理论,在声明j的时候,会去栈中招有没有字面值为2的地址,由于在栈中已经有2这个字面值,便将j直接指向2的地址。这样,就出现了i与j同时均指向2的情况。

拿出javap -c进行反编译:结果如下:

2b413bb5f041db5fcc7a163f37cc201d.png

Code:0: iconst_2 //把2放到栈顶

1: istore_1 //把栈顶的值放到局部变量1中,即i中

2: iconst_2 //把2放到栈顶

3: istore_2 //把栈顶的值放到局部变量2中,即j中(i 和 j同时指向2)

4: return

虽然这里说i和j同时指向2,但这里不等于说i和j指向同一块地址(java是不允许程序员直接修改堆栈中的数据的,所以就不要想着,我是不是可以修改栈中的2,那样岂不是i和j的值都会变化。另:在编译器内部,遇到j=2;时,它就会重新搜索栈中是否有2的字面值,如果没有,重新开辟地址存放2的值;如果已经有了,则直接将j指向这个地址。因此,就算j另被赋值为其他值,如j=4,j值的改变不会影响到i的值。)

再来一个例子:

adef822d6381c2241d8f46ec0b2526c1.png

还是javap -c

20f4abb644239129dc10a65c4028967e.png

Code:0: iconst_2 //把2放到栈顶

1: istore_1 //把栈顶的值放到局部变量1中,即i中

2: iload_1 //把i的值放到栈顶,也就是说此时栈顶的值是2

3: istore_2 //把栈顶的值放到局部变量2中,即j中

4: return

看到这里是不是有点明确了?

既然我们对javap有了一定的了解,那我们就开始用它来解决一些实际的问题:

3.i++和++i的问题

53fc33fa8641b0b3d6a2dd484dddf094.png

反编译结果为

9f5bb677fa55179ccfde06738542625c.png

Code:0: iconst_11: istore_12: iinc 1, 1 //这个个指令,把局部变量1,也就是i,增加1,这个指令不会导致栈的变化,i此时变成2了

5: iconst_16: istore_27: iinc 2, 1//这个个指令,把局部变量2,也就是j,增加1,这个指令不会导致栈的变化,j此时变成2了

10: return

可以看出,++在前在后,在这段代码中,没有任何不同。

我们再看另一段代码:

bcb2afd7778da07f9d4a624dd55adea6.png

反编译结果:

6eae7271674dee21f3559bfc707a4f70.png

Code:0: iconst_11: istore_12: iload_13: iinc 1, 1 //局部变量1(即i)加1变为2,注意这时栈中仍然是1,没有改变

6: istore_1 //把栈顶的值放到局部变量1中,即i这时候由2变成了1

7: iconst_18: istore_29: iinc 2, 1 //局部变量2(即j)加1变为2,注意这时栈中仍然是1,没有改变

12: iload_2 //把局部变量2(即j)的值放到栈顶,此时栈顶的值变为2

13: istore_2 //把栈顶的值放到局部变量2中,即j这时候真正由1变成了2

14: return

是否看明白了? 如果这个看明白了,那么下面的一个问题应该就是迎刃而解了:

4b6e36f9588e320b64cdeab230235b8a.png

m = m ++;这句话,java虚拟机执行时是这样的: m的值加了1,但这是栈中的值还是0, 马上栈中的值覆盖了m,即m变成0,因此不管循环多少次,m都等于0。

如果改为m = ++m; 程序运行结果就是100了。。。

案例2

public classTestJavap {public static voidmain(String[] args) {int i=0;

i= i++ + ++i;

System.out.println(i);

}

}

$ javac TestJavap.java

$ javap-c TestJavap

Compiled from"TestJavap.java"

public class TestJavap extendsjava.lang.Object{publicTestJavap();

Code:0:  aload_01:  invokespecial  #1; //Method java/lang/Object."":()V

4:  return

public static voidmain(java.lang.String[]);

Code:0:  iconst_01:  istore_12:  iload_13:  iinc    1, 16:  iinc    1, 19:  iload_110:  iadd11:  istore_112:  getstatic      #2; //Field java/lang/System.out:Ljava/io/PrintStream;

15:  iload_116:  invokevirtual  #3; //Method java/io/PrintStream.println:(I)V

19:  return

解释如下

int i = 0;

i=i++ + ++i;

Code:

0:   iconst_0        将 0 推到堆栈中//对应赋值语句 int i = 0;iconst_0中的0为初始值

1:   istore_1        从堆栈中弹出这个值,并将它存储到局部变量表的索引 1 处 。对应赋值语句,上句是赋值,这句是存储,索引1处即为i,因为没有其它变量,所以本例中局部变量的索引不变

2:   iload_1         将局部变量表索引 1 处的值推到堆栈中。将局部变量索引1处的计算结果推入堆栈临时存储,局部变量的索引从1开始,依此类推,这句意味着将变量i=0先推入堆栈临时存储

3:   iinc    1, 1       局部变量表索引 1 处的变量加 1 。将变量i自加1,则此时变量i为1

6:   iinc    1, 1       局部变量表索引 1 处的变量加 1 。再将变量i自加1,则此时变量i为2

9:   iload_1           将局部变量表索引 1 处的值推到堆栈中。将变量i=2推入堆栈临时存储

10:  iadd               从操作数堆栈中弹出两个整数并让它们相加。将得到的整数推回堆栈中//将两次暂存的整数相加,即0+2=2

11:  istore_1        从堆栈中弹出这个值,并将它存储到局部变量表的索引 1 处。把上步相加之结果弹出堆栈

12:  getstatic       #2; //Field java/lang/System.out:Ljava/io/PrintStream;  对应System.out

15:  iload_1            加载i//将变量i=2推入堆栈临时存储

16:  invokevirtual   #3; //Method java/io/PrintStream.println:(I)V            对应println

19:  return             退出方法

案例3

public classTest01 {public static voidmain(String[] args) {int a = 1;int b = 1;int c = a +b;

}

}

2af56fb6225c76b15955f10f45d46d2e.png

E:\namespace\test01\src\com\first\ock>javap -c Test01

警告: 二进制文件Test01包含com.first.ock.Test01

Compiled from"Test01.java"

public classcom.first.ock.Test01 {publiccom.first.ock.Test01();

Code:0: aload_01: invokespecial #1 //Method java/lang/Object."":()V

4: return

public static voidmain(java.lang.String[]);

Code:0: iconst_11: istore_12: iconst_13: istore_24: iload_15: iload_26: iadd7: istore_38: return}

参考文章:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/360203.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

hibernate 映射四多对一双向映射

学生和班级的例子来说,多个学生可以对应一个班级。 1.站在学生角度来说,学生就是多端。那么我们可以通过多对一来维护2者的关系。 2.站在班级的角度来说,班级就是一端。那么我们可以通过一对多啦维护2者的关系。 3.我们也可以双向关联两者的关…

django mysql 表单_Python Django 表单提交数据到mysql并展示

首先1: 新建项目userproject, 新建应用childName2: 这是childName文件目录,templates文件夹放insert.html 与 show.html3: insert.html 与 show.html/**insert.html**/用户登录12345678910111213141516Title信息展示用…

体验最火的敏捷——SCRUM(厦门,2014.1.4)

1.概述SCRUM是当前最火的一种敏捷开发方法,有用户故事、冲刺、燃尽图等很多很酷的玩法,有牛B的产品负责人、SCRUM Master,有超强的自组织团队。本沙龙将为您展现当前最火最酷的敏捷开发方法!内容大纲:1)SCRUM是神马东西…

[单选]物联网产业链的主要产品不包括下列哪一项 - 关于物联网(主讲:柳毅)笔记...

[单选]物联网产业链的主要产品不包括下列哪一项 转载于:https://www.cnblogs.com/scgw/p/3488452.html

java web读取excel_JavaWeb使用POI操作Excel文件实例

1.为项目添加POI点进去之后下载(上边的是编译好的类,下边的是源代码)解压文件夹,把下面三个文件复制到WebComtent>WEB-INF>lib文件夹下再把这三个文件复制到Tomcat的lib文件夹下,否则Tomcat会因为找不到类而报错(这个地方郁闷了一上午)…

java 静态方法与实例方法的区别_静态方法与实例方法的区分

Java系列之:看似简单的问题 静态方法和实例化方法的区别Java程序启动class文件被读取时类被加载,如果有static方法,此时会分配内存,非static方法实例化类时才在内存中分配控件存储,引用存储在堆栈中,实体存…

easyui源码翻译1.32--ValidateBox(验证框)

前言 使用$.fn.validatebox.defaults重写默认值对象。下载该插件翻译源码 validatebox(验证框)的设计目的是为了验证输入的表单字段是否有效。如果用户输入了无效的值,它将会更改输入框的背景颜色,并且显示警告图标和提示信息。该验证框可以结合form(表单…

学java前要学css_教你一招:学习Java必须学会的CSS用法

一:CSS概述什么是CSSCSS就是层叠样式表(Casading Style Sheets),通常称为CSS样式表,或者是级联样式表。主要用于设置HTML中的文本,内容(字体,大小,对齐),图片外形(宽高,边框样式&…

标准I/O库之缓冲

标准I/O库提供缓冲的目的是尽可能减少使用read和write调用的次数。它也对每个I/O流自动地进行缓冲管理,从而避免了应用程序需要考虑这一点所带来的麻烦。 标准I/O提供了三种类型的缓冲: (1)全缓冲。这种情况下,在填满标…

java如何写安卓接口文档_android、java制作sdk以及自动生成文档

最近一直在做android开发,昨天经理让我写个接口SDK做个接口文档,以便后面的开发。这让我很焦灼,SDK怎么做?要是只有敲代码还好。可是那个接口文档!!!文档这东西最讨厌了,头都大了后来…

21世纪的设计模式:抽象工厂模式

这是我的演讲的第二部分,“ 21世纪的设计模式” 。 此模式在Java代码中到处都有使用,尤其是在更多“企业”代码库中。 它涉及一个接口和一个实现。 该界面如下所示: public interface Bakery {Pastry bakePastry(Topping topping);Cake bak…

java动态变量名反射_Java动态性—反射 - Eclipse666的个人空间 - OSCHINA - 中文开源技术交流社区...

1.什么是动态语言?程序运行时,可以改变程序的结构或者变量类型;如Python,javaScriptfunction(){var s"var a3;var b4;" evals(s);}在执行javascript代码的的过程中,可以改变变量的值或插入语句改变结构。但J…

mysql构建栋_【转载】这次拆库 应是微服务化的拆分方式

一、现状现状.png我们将一个大而全的系统一拆为三,容器,发布,测试都已经独立出去,但是原始的数据库还是一套,现在需要将数据库做一个拆分,A、B、C三个系统有各自的数据库之后,我们的微服务化在现…

使用SharePoint 2010新增的文档集内容类型来管理文档

使用SharePoint 2010新增的文档集内容类型来管理文档 SharePoint 2010新增加的文档集功能是作为内容类型存在的,使用范围在网站集中,需要激活“文档集”功能到网站集才可以正常使用。文档集其实就是一个SharePoint产品增强的文件夹和内容类型的综合体现。…

基于visual Studio2013解决算法导论之019栈实现(基于数组)

&#xfeff;&#xfeff;&#xfeff;题目用数组实现栈解决代码及点评#include <stdio.h> #include <stdlib.h> #include <time.h> #include <malloc.h> #include<assert.h>typedef struct Stack {int nTop;int nLen;int *pnArr; }Stack, *PSta…

关闭终端php就退出进程_解决windows下php-cgi进程经常自动关闭

php-cgi在linux中有fpm管理&#xff0c;Apache不是采用cgi这种模式&#xff0c;于是乎就出现了&#xff0c;在windows下&#xff0c;采用nginx时&#xff0c;开启的php-cgi监听非常不稳定&#xff0c;经常奔溃关闭。现在解决了这个问题。注意是win32位系统下解决的&#xff0c;…

php为什么获取不到id,微信小程序无法获取到unionId怎么办

UnionID机制说明如果开发者拥有多个移动应用、网站应用、和公众帐号(包括小程序)&#xff0c;可通过unionid来区分用户的唯一性&#xff0c;因为只要是同一个微信开放平台帐号下的移动应用、网站应用和公众帐号(包括小程序)&#xff0c;用户的unionid是唯一的。换句话说&#x…

CUBA平台–新的Java企业应用程序框架

所以..你好&#xff0c;世界&#xff01; 我们的英语网站终于可以正常使用了&#xff0c;现在每个人都可以下载该平台&#xff0c;并可以以前所未有的速度更快地创建业务应用程序。 在我们决定与国际Java社区共享足够好之前&#xff0c;我们花了六年的永久发展和偶尔的革命&…

Abiword页面布局

Abiword页面布局 AP_Win32FrameImpl::_DocumentWndProc 文档窗口过程函数在WM_SIZE消息中设置FV_View对象的整体尺寸&#xff0c;跟窗体的&#xff1a;设备单位 转换成&#xff1a;布局单位。m_iWindowWidth 19695&#xff0c;m_iWindowHeight 8520 设备单位和布局单位的比例是…

php 文件内容对比,php 比较两个文件是否相同

-### php比较两个文件是否相同小一点的文件直接实用内置函数就可以了function md5_files($filename,$filename1){$file1 md5_file($filename);$file md5_file($filename1);if($file $file1){return "文件相同";}else{return "文件不同";}}md5_file()…