解析类和对象的初始化过程物业
中药养生 2021年09月06日 浏览:2 次
在与初始化时机相关的类装载时机问题上,Java虚拟机规范并没有对其做严格的定义,这就使得JVM在实现上可以根据自己的特点提供采用不同的装载策略。我们可以思考一下Jboss AOP框架的实现原理,它就是在对你的class文件装载环节做了手脚--插入了AOP的相关拦截字节码,这使得它可以对程序员做到完全透明化,哪怕你用new操作符创建出的对象实例也一样能被AOP框架拦截--与之相对应的Spring AOP,你必须通过他的BeanFactory获得被AOP代理过的受管对象,当然Jboss AOP的缺点也很明显--他是和JBOSS服务器绑定很紧密的,你不能很轻松的移植到其它服务器上。嗯~……,说到这里有些跑题了,要知道AOP实现策略足可以写一本厚厚的书了,嘿嘿,就此打住。
说了这么多,类的初始化时机就是在“在首次主动使用时”,那么,哪些情形下才符合首次主动使用的要求呢?
首次主动使用的情形:
◆创建某个类的新实例时--new、反射、克隆或反序列化;
◆调用某个类的静态方法时;
◆使用某个类或接口的静态字段或对该字段赋值时(final字段除外);
◆调用Java的某些反射方法时;
◆初始化某个类的子类时;
◆在虚拟机启动时某个含有main()方法的那个启动类。
除了以上几种情形以外,所有其它使用JAVA类型的方式都是被动使用的,他们不会导致类的初始化。
我的问题究竟出在哪里
好了,了解了JVM的类初始化与对象初始化机制后,我们就有了理论基础,也就可以理性的去分析问题了。
下面让我们来看看前面[清单一]的JAVA源代码反组译出的字节码:
[清单三]
public class chingEnumResolver ject{
static {};
Code: 0: new #2;
//class CachingEnumResolver
3: dup
4: invokespecial #14;
//Method \"init\":()V ①
7: putstatic #16;
//Field SINGLE_ENUM_RESOLVER:Lcom/ccb/framework/enums/CachingEnumResolver;
10: new #18;
//class HashMap ②
13: dup
14: invokespecial #19;
//Method java/util/HashMap.\"init\":()V
17: putstatic #21;
//Field CODE_MAP_CACHE:Ljava/util/Map;
20: getstatic #21;
//Field CODE_MAP_CACHE:Ljava/util/Map;
而是我们坐在一起 23: ldc #23;
//String 0
25: ldc #25;
//String 北京市
27: invokeinterface #31, 3;
//InterfaceMethod java/util/t:(Ljava/lang/Object;Ljava/lang/Object;)
Ljava/lang/Object; ③
32: pop 33: returnprivate chingEnumResolver();
Code: 0: aload_0 1: invokespecial #34;
//Method java/lang/Object.\"init\":()V 4: invokestatic #37;
//Method initEnums:()V ④ 7: returnpublic static void initEnums();
Code: 0: getstatic #21;
//Field CODE_MAP_CACHE:Ljava/util/Map;
⑤ 3: ifnonnull 24 6: getstatic #44;
//Field java/lang/t:Ljava/io/PrintStream;
9: ldc #46;
//String CODE_MAP_CACHE为空,问题在这里开始暴露.
11: invokevirtual #52;
//Method java/io/intln:(Ljava/lang/String;)V 14: new #18;
//class HashMap 17: dup 18: invokespecial #19;
//Method java/util/HashMap.\"init\":()V ⑥ 21: putstatic #21;
//Field CODE_MAP_CACHE:Ljava/util/Map;
24: getstatic #21;
//Field CODE_MAP_CACHE:Ljava/util/Map;
27: ldc #54;
//String 1 29: ldc #25;
//String 北京市 31: invokeinterface #31, 3;
//InterfaceMethod java/util/t:(Ljava/lang/Object;
Ljava/lang/Object;)Ljava/lang/Object;
⑦ 36: pop 37: getstatic #21;
//Field CODE_MAP_CACHE:Ljava/util/Map;
40: ldc #56;
//String 2 42: ldc #58;
//String 云南省 44: invokeinterface #31, 3;
//InterfaceMethod java/util/t:(Ljava/lang/Object;Ljava/lang/Object;)
Ljava/lang/Object;
⑧ 49: pop 50: returnpublic p getCache();
Code: 0: getstatic #21;
//Field CODE_MAP_CACHE:Ljava/util/Map;
3: invokestatic #66;
//Method java/util/modifiableMap:(Ljava/util/Map;)Ljava/util/Map;
6: areturnpublic static chingEnumResolver getInstance();
Code: 0: getstatic #16;
//Field SINGLE_ENUM_RESOLVER:Lcom/ccb/framework/enums/CachingEnumResolver;
⑨ 3: areturn}
如果上面[清单一]显示,清单内容是在JDK1.4环境下的字节码内容,可能这份清单对于很大部分兄弟来说确实没有多少吸引力,因为这些JVM指令确实不像源代码那样漂亮易懂。但它的的确确是查找和定位问题最直接的办法,我们想要的答案就在这份JVM指令清单里。
现在,让我们对该类从类初始化到对象实例初始化全过程分析[清单一]中的代码执行轨迹。
治白癜风那家医院专业南通有专业白癜风医院吗
健脾胃的中药有哪些

- 上一篇: 解析类和对象的初始化过程
- 下一篇 解析类和对象的初始化过程计划
-
柿霜的炮制方法
2019-07-16
-
麦门冬的炮制方法
2019-07-15
-
中医推荐瘦脸良方颜面针灸
2019-07-15
-
能老了不掉牙的秘诀
2019-07-13
-
2017全国中医药现代化论坛举办
2019-07-12
-
牛乳粥增白窍门
2019-07-06