如何在Linux中反编译java源代码
怎么⽤java⾃带反编译⼯具查看java字节码⽂件图⽂教程,⽤java⾃带反编译⼯具查看java字节码⽂件,⾸先,配置好java开发环境,环境变量各种配置,这⾥就不⼀⼀说了,相信⼤家都配置好了。下⾯⼊正题----怎么⽤java⾃带反编译⼯具查看java字节码⽂件
⼯具/原料
javap
java+java开发⼯具
⽅法/步骤
1. ⾸先,写⼀个最简单的程序,我们从最简单⼊⼿,相信⼤家1秒就可以看懂了吧
2. 然后,经过编译⽣成HelloWorld.class⽂件,对,我们就去看HelloWorld.class⾥⾯的java字节码,到底是什么。
输⼊下图所⽰的命令,javap -verbose HellloWorld 回车,⼊下图所⽰。
3. 详细代码,如下,没学过编译原理的,建议去学下,去学下编译原理,经常看看,就能看懂了,我和⼤家都是⼀样,我也是⼀名,计
算机爱好者,没事,喜欢瞎⿎弄。今天就到这了。怎么⽤java⾃带反编译⼯具查看java字节码⽂件
END
注意事项
怎么⽤java⾃带反编译⼯具查看java字节码⽂件图⽂教程
怎么⽤java⾃带反编译⼯具查看java字节码⽂件图⽂教程
⼀直在学习Java,碰到了很多问题,碰到了很多关于i++和++i的难题,以及最经典的String str = "abc" 共创建了⼏个对象的疑难杂症。 知道有⼀⽇知道了java的反汇编 命令 javap。现将学习记录做⼀⼩结,以供⾃⼰以后翻看。如果有错误的地⽅,请指正
java技术介绍百度百科1.javap是什么:
where options include:
-
c Disassemble the code
-classpath <pathlist> Specify where to find user class files
-extdirs <dirs> Override location of installed extensions
-help Print this usage message
-J<flag> Pass <flag> 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 <pathlist> 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
从⼀个最简单的例⼦开始:
这个例⼦中,我们只是简单的声明了两个int型变量并赋上初值。下⾯我们看看javap给我们带来了什么:(当然执⾏javap命令前,你得⾸先配置好⾃⼰的环境,能⽤javac编译通过了,即:javac TestJavap.java )
我们只看(⽅便起见,将注释写到每句后⾯)
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的地址。
看了这段话,再⽐较下上⾯的注释,是不是完全吻合?
为了验证上⾯这⼀说法,我们继续实验:
我们将 i 和 j的值都设为2。按照以上理论,在声明j的时候,会去栈中招有没有字⾯值为2的地址,由于在栈中已经有2这个字⾯值,便将j直接指向2的地址。这样,就出现了i与j同时均指向2的情况。
拿出javap -c进⾏反编译:结果如下:
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的值。)
再来⼀个例⼦:
还是javap -c
Code:
0: iconst_2 //把2放到栈顶
1: istore_1 //把栈顶的值放到局部变量1中,即i中
2: iload_1 //把i的值放到栈顶,也就是说此时栈顶的值是2
3: istore_2 //把栈顶的值放到局部变量2中,即j中
4: return
看到这⾥是不是有点明确了?
既然我们对javap有了⼀定的了解,那我们就开始⽤它来解决⼀些实际的问题:1.i++和++i的问题
反编译结果为
Code:
0: iconst_1
1: istore_1
2: iinc 1, 1 //这个个指令,把局部变量1,也就是i,增加1,这个指令不会导致栈的变化,i此时变成2了 5: iconst_1
6: istore_2
7: iinc 2, 1//这个个指令,把局部变量2,也就是j,增加1,这个指令不会导致栈的变化,j此时变成2了 10: return
可以看出,++在前在后,在这段代码中,没有任何不同。
我们再看另⼀段代码:
反编译结果:
Code:
0: iconst_1
1: istore_1
2: iload_1
3: iinc 1, 1 //局部变量1(即i)加1变为2,注意这时栈中仍然是1,没有改变
6: istore_1 //把栈顶的值放到局部变量1中,即i这时候由2变成了1
7: iconst_1
8: istore_2
9: iinc 2, 1 //局部变量2(即j)加1变为2,注意这时栈中仍然是1,没有改变
12: iload_2 //把局部变量2(即j)的值放到栈顶,此时栈顶的值变为2
13: istore_2 //把栈顶的值放到局部变量2中,即j这时候真正由1变成了2
14: return
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论