zoukankan      html  css  js  c++  java
  • 为什么匿名内部类只能访问其所在方法中的final类型的局部变量?

      大部分时候,类被定义成一个独立的程序单元。在某些情况下,也会把一个类放在另一个类的内部定义,这个定义在其他类内部的类就被称为内部类,包含内部类的类也被称为外部类。

    class Outer {
    
        private int a;
    public class Inner { private int a; public void method(int a) { a++;      // 局部变量 this.a++;   // Inner类成员变量 Outer.this.a++; // Outer类成员变量 } } }

      对于上面的成员内部类,一般做法是在Outer中写一个返回Inner类对象的方法

    public Inner getInner() {
        return new Inner();
    }

      在其他类中使用该成员内部类:

    Outer outer = new Outer();
    Outer.Inner inner = outer.getInner();
    // 或者Outer.Inner inner = outer.new Inner();

      而对于静态内部类,就不需要创建外部类的实例了:

    Outer.StaticInner inner = new Outer.StaticInner();

      匿名内部类不能访问外部类方法中的局部变量,除非该变量被声明为final类型

      1. 这里所说的“匿名内部类”主要是指在其外部类的成员方法内定义的同时完成实例化的类,若其访问该成员方法中的局部变量,局部变量必须要被final修饰。原因是编译器实现上的困难:内部类对象的生命周期很有可能会超过局部变量的生命周期
      2. 局部变量的生命周期:当该方法被调用时,该方法中的局部变量在栈中被创建,当方法调用结束时,退栈,这些局部变量全部死亡。而内部类对象生命周期与其它类对象一样:自创建一个匿名内部类对象,系统为该对象分配内存,直到没有引用变量指向分配给该对象的内存,它才有可能会死亡(被JVM垃圾回收)。所以完全可能出现的一种情况是:成员方法已调用结束,局部变量已死亡,但匿名内部类的对象仍然活着。
      3. 如果匿名内部类的对象访问了同一个方法中的局部变量,就要求只要匿名内部类对象还活着,那么栈中的那些它要所访问的局部变量就不能“死亡”。
      4. 解决方法:匿名内部类对象可以访问同一个方法中被定义为final类型的局部变量。定义为final后,编译器会把匿名内部类对象要访问的所有final类型局部变量,都拷贝一份作为该对象的成员变量。这样,即使栈中局部变量已经死亡,匿名内部类对象照样可以拿到该局部变量的值,因为它自己拷贝了一份,且与原局部变量的值始终保持一致(final类型不可变)。

      最后,Java 8更加智能:如果局部变量被匿名内部类访问,那么该局部变量相当于自动使用了final修饰

  • 相关阅读:
    枚举类
    泛型数组列表
    方法参数
    给一个数字和列表,判断列表中是否存在两个元素之和等于这个数字,并好之两个数的坐标
    selenium.common.exceptions.ElementNotVisibleException: Message: element not interactable报错
    python-selenium提供的execute_script调用js操作
    xlrd读取excel数据封装
    0531-练习题 os.system
    0528 文件操作习题
    05/17
  • 原文地址:https://www.cnblogs.com/eniac12/p/5240100.html
Copyright © 2011-2022 走看看