与虚拟机加载器有关,对象的加载与销毁

 

一  Java和C++的一个显著不同之处在于,它的对象是动态加载的,C++无法模仿。

      JVM有一个组成部分--原生类加载器。它只加载可信类(比如Java API类库)。如果有WebStart类型的应用需求,或者企业级应用(RMI通信等等),则可能会在加载器链条上放上一些别的加载器。

 

二     什么时候类会被加载呢?

     当一个类第一次被使用的时候,它最早可以追溯到对它的静态成员的第一次引用的时候,它就会被加载。由此可以推理:

     1 实际上构造器也是一个静态成员,虽然它没有使用static关键字。如果做个试验我们就会发现,即使一个没有任何static成员的类中,它在初始化的时候,还是要先初始化父类的构造器--这就好像父类的构造器是一个静态成员一样。所以实际上所有的构造器都是静态成员。

 

     2 因为所有的静态成员都是一次性初始化的,所以初始化顺序为

     父类静态成员-父类静态方法块 - 本类静态成员 - 本类静态方法块(也就是说,加载过程中就有static初始化了,但类本身在堆上的无数成员还没有刷新过。从思想上看,静态聚在一起,归类,成员归在一起,归对象。)-父类非静态成员 - 父类非静态方法- 父类构造器 -  本类非静态成员 - 本类非静态方法块 -本类构造器。

       A static

       InitializationSequence static

       A member

       InitializationSequence member

 

      其中有个小技巧就是用{}包围的非静态初始化块也可以执行代码!但不用{}包围就不可以!

      值得注意的是,类在生命引用的时候就被加载了,但只有当对静态成员进行引用(更直接地说,访问)的时候,他们才被初始化。此外:

     1 除非强制访问编译时常量 final static,否则这种编译时展开的静态常量不算一种初始化,不会引发其他的初始化。

     2 任何一个对于静态成员的访问,都必须等到所有静态初始化结束以后才能开始。比如我在一个static的变量a前写一个static的print方法b,在我打印a的时候,实际上b先打印。

 

 

 

 

     为了类的使用而做的准备工作实际上包含三个步骤:

 

    1 加载 用字节码生成一个Class对象。

    2 链接 验证字节码,为静态存储区域分配内存空间。

    3 初始化 按照初始化的顺序初始化。实际上初始化放在第三步,相当于被推迟了。

 

三、构造器的作用

 

      构造器总是最后执行,以确保成员初始化式结束以后还能正确地收尾。所以成员初始化式很重要,它能保证不同的构造器拥有相同的成员初值加以变化,可以健壮地构造一个基准对象在内存上的成员变量区间以支持不同的构造器。

       我们应该注意到,构造器都是static的,本来静态方法不可以引用成员方法的,但在构造器中可以。即使构造器没有执行完,半途中堆上也已经有一个对象了,这个堆上的对象可以运用成员方法了。但这是很危险的行为,因为执行到半途的对象调用成员方法可能涉及没有充分初始化的对象成员,容易出现运行时异常。

 

 

 

四、为什么编译器会强制地调用基类的构造器而子类的终结器必须手动组成调用链?

 

 

      任何一个对象在构造完成的时候,都必须保证所有的成员是可用的。而基类的成员变量对于子类而言几乎是不用见的。要保证这个对象的初始化成功,就必须要递归地调用基类的构造器。所以编译器会强制调用,干脆帮我们串起来。除非我们手动指定父类的编译器(这必须是构造器中第一句话),否则都默认调用缺省构造器。这提醒我们一个问题,即每个基类都要有一个缺省构造器。

       但终结可能是一个特殊的行为。构造器在堆上做托管成员(容我从C#中借来这个词)的初始化在完全的链式反应下是不会出任何差错的。但终结器要做的很多的是非托管的资源的释放与终结,直接一层层调用父类终结器就可能出问题。于是出现两种策略。一,在本对象的终结器中完成一切工作,这就要求本对象将终结器函数设为虚函数,这看起来似乎是理所当然的,但C++居然不是这样做的!C++的析构函数是非虚的普通函数!二,就是在处理完本对象特有的资源以后,在最后一局手动调用父类的终结器方法。千万不要干多余的事情,重复地释放一个资源有可能引发灾难性的后果。

 

 

注:关于初始化顺序请看

import static net.mindview.util.Print.*;

public class Base {

	/**
	 * @param args
	 */
	static
	{
		print(1);
	}
	int a = 3;
	static double b = 1.5;
	Base()
	{
		print(a);
	}
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		print(Derived.b); 
		Derived d = new Derived();
		print("final");
	}

}
 class Derived extends Base
 {
	 int a = 4;
	 static 
	 {
		 print(2);
	 }
	 
	 Derived()
	 {
		 print(a);
	 }
	 
 }

 

 

结果为:

 

1
1.5
2
3
4
final
 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值