问题:
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
image = (ImageView) findViewById(R.id.image);
final ClipDrawable clip = (ClipDrawable) image.getBackground();
final Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
clip.setLevel(clip.getLevel() + 200);
super.handleMessage(msg);
}
};
final Timer timer = new Timer();
timer.schedule(new TimerTask() {
@Override
public void run() {
if (clip.getLevel() < 10000) {
handler.sendEmptyMessage(0);
} else {
timer.cancel();
}
}
}, 0, 300);
}
我的理解是:局部变量的生命周期是在{}之间
我的疑问是:clip变量和timer的生命周期也在{}之间,那么当这个onCreate方法执行完成以后
JVM不是应该将这两个局部变量从Stack内存中清除吗,这时候timer应不存在了,在timer.cancel(),不是应该报空指针吗?为什么不报呢?
帖子1
你的timerTask是一个inner class,java规定inner class要访问clip和timer这两个local variable,必须把local variable设置成final。这样你的inner class就知道在整个Activity的生命周期中,clip和timer的值是不会变化的,不会被deallocate掉,可以放心使用。local variable加上final关键字后,和member variable的生命周期是一样的。其实和在Activity类中声明成两个member variable也几乎是一样的。换言之,一旦一个局部变量被final修饰了,那么他的生命周期和外部类对象的生命周期是一样的了。
如例中所示,在外部类Outer中声明了一个内部类TimerPrint,这个类中的方法引用了方法start中的一个局部变量testTxt
逻辑上:因为该内部类出现在一个方法的内部,但实际编译时,内部类编译为Outer$1TimerPrint.class,这说明,外部类的这个方法和内部类是处于同一级别的。换句话说是非final变量和内部类的生命周期不一样!start被调用后,非final变量也会随之消失,就会出现内部类引用非法!
实际做法:java编译器的行为是这样的(前提条件是该变量在内部类中被引用):
若定义为final,则java编译器则会在内部类TimerPrint内生成一个外部变量的拷贝,而且可以既可以保证内部类可以引用外部属性,又能保证值的唯一性。
若不定义为final,则无法通过编译!(jdk1.6测试过)。因为编译器不会给非final变量进行拷贝,那么内部类引用的变量就是非法的!
下面看经过编译以后的字节码:
如果外部类中的变量d没有被内部类引用,则final为可选的,而且java编译器将不做特殊处理!!
加一个参数d 并且把它定义为非final类型,编译以后文件如下:
由上可以看出,在方法内部定义内部类时,内部类如果调用了方法内的变量,则该变量必须被final修饰,否则就会因为在调用内部类时因为找不到所用的变量而报错!