作者darkk6 (Mr. Pan)
看板java
标题Re: [问题] static runtime问题
时间Sun Dec 21 13:48:18 2014
1F:推 icydream: 我比较好奇的是,既然static member不需要instance,就12/21 12:34
2F:→ icydream: 可以呼叫使用,那为啥在撰写的时候,却可以使用instance12/21 12:36
3F:→ icydream: 来呼叫呢?(有warning,但不影响执行)12/21 12:37
PS : 底下都是在 Oracle 的 JDK 的结果,其他 Compiler 会不会帮你做 我就没研究了
这个其实和 AutoBoxing 一样,是 Java Compiler 帮你做的事情
Keyword : Compiler Sugar 别只顾着吃糖阿...
这边要注意到一点,就是你写的 Code 并不是 100% 按照 Source Code
编译成 .class 档案,直接举 AutoBoxing 来说,你写的 Code 是这样:
Integer a=1;
经过编译器的时候先帮你 Boxing 变成
Integer a = Integer.valueOf(1);
所以实际写进 .class 的内容是
04 B8
00 02 4C B1
绿色的部分不一定,那是 method ref. index
PS : 参考 Java Byte Code
http://en.wikipedia.org/wiki/Java_bytecode_instruction_listings
04 = iconst_1 // 将常数 1 放进 stack
B0
00 02 = invokestatic
#2 // 将 stack 的职传入 Method No2 执行
// 在这里 Method No2 是 Integer.valueOf
C4 = astore_1 // 将 reference 储存到编号1的变数
B1 = return 是因为我的 main 里面只有这一行,所以到底了要 return
参考图片:
http://ppt.cc/Fte5
----------
回到正题,为何 static field 可以透过 instance 呼叫,只会出现警告而
不会有错误呢? 因为编译器也帮你找到你的 instance 是哪个 class 在编译时
转换好了:
PS : 重申,我只能确定 Oracle 的 JDK 会帮你做
请直接参考图片:
http://ppt.cc/6bZX
左边是 Source Code , 中间是 ByteCode
右边是根据 ByteCode decompile 回来的 Source Code
在 .java 先使用 TEST.StaticInt=1 存取 static field
接着再用一个 TEST 的实体 t 存取 StaticInt
为了比较,另外也存取了一个 InstanceInt
在 Byte Code 的地方可以看到底下
t.StaticInt=2 和
t.InstanceInt=2
所使用的 opcode 不同
一个是 putfield (对 InstanceInt)
一个是 putstatic (对 StaticInt)
而根据这个 byte code 编译回来的 source code (右边)
也可以看出其实编译器在编译的时候,已经帮你把 t.StaticInt=2
转成 TEST.StaticInt 了
至於为什麽会提出警告嘛... 我猜应该是怕你自己会搞不清楚吧
举个例子,猜猜看这样会印出啥呢?
http://pastie.org/9792378
解答:
http://ideone.com/sZ3aXb ( 印象中 SCJP 会考这种东西 )
--
※ 发信站: 批踢踢实业坊(ptt.cc), 来自: 223.138.115.115
※ 文章网址: http://webptt.com/cn.aspx?n=bbs/java/M.1419140902.A.C90.html
※ 编辑: darkk6 (223.138.115.115), 12/21/2014 14:00:33
4F:推 icydream: 感谢,获益良多 12/21 14:12
5F:→ darkk6: 发现我的 Son 打错了 ..... 变太阳了 O.O 12/21 14:32