<span style="font-size:18px;">class Creature {

private static long numCreated = 0;

public Creature() {
numCreated++;
}

public static long numCreated() {
return numCreated;
}
}

public class Creator {

public static void main(String[] args) {

for (int i = 0; i < 100; i++)
Creature Creature = new Creature();

System.out.println(Creature.numCreated());
}
}</span>

本地变量声明问题小结:


(1)上面的Java代码看起来似乎应该打印100,但是实际上它什么也没有打印!


(2)通过Javac编译可以发现一些错误的诊断信息,但对于解决问题作用不大。



问题剖析:

1、Java语言规范不允许一个本地变量声明语句在for、while或do循环中重复执行。


2、一个本地变量声明作为一条语句只能直接出现在一个语句块中。


3、一个语句块是由一对花括号以及包含这对花括号中的语句和声明构成的。



解决方法:

1、将Creature creature = new Creature 这个声明置于语句块中。


2、将该声明用一个无任何修饰的构造器调用(new Creature)来替代将更有实际意义,这样强调对新创建对象的引用正在被丢弃。


再问题抛出:

1、用于跟踪Creature实例个数的变量(numCreated)是long类型而不是int类型。原因是:我们很容易想到,一个城程序创建出的某个实例可能多于int数值的最大值,但是它不会多于long数值的最大值。int数值最大值为2^31-1,long数值的最大值为2^63-1.


温馨提醒:

1、题目中创建的计数策略并不是线程安全的。如果多个线程可以并行地创建对象,那么递增计数器的代码和读取计数器的代码都应该被同步。(synchronized)

2、代码块的改写如下所示:


class Creature {

private static long numCreated = 0;

public Creature() {
synchronized (Creature.class){
numCreated++;
}
}

public static synchronized long numCreated() {
return numCreated;
}
}


3、另外,如果使用的是5.0或者更新的版本,可以使用一个AtomicLong实例,它在面临并发时可以绕过对同步的需求。

4、代码事例:


import java.util.concurrent.atomic.AtomicLong;

public class Creature {

private static AtomicLong numCreated = new AtomicLong();

public Creature(){
numCreated.incrementAndGet();
}

public static long numCreated(){
return numCreated.get();
}
}



总结:


(1)一个本地变量声明不能被用作for、while、do循环中重复执行语句,它作为一条语句只能出现在一个语句块汇总。

(2)在使用一个变量来对实例的创建进行计数时,要使用long类型而不是int类型的变量,以防止溢出!

(3)如果打算在多线程中创建实例,要么将对实例计数器的访问进行同步,要么使用一个AtomicLong类型的计数器。