逝者如斯,不舍昼夜

尘世中一个迷途小书童,读书太少,想得太多
随笔 - 72, 文章 - 0, 评论 - 60, 阅读 - 83万
  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理
< 2025年3月 >
23 24 25 26 27 28 1
2 3 4 5 6 7 8
9 10 11 12 13 14 15
16 17 18 19 20 21 22
23 24 25 26 27 28 29
30 31 1 2 3 4 5

 

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

复制代码
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. 局部变量的生命周期:当方法被调用时,其内部的局部变量在栈中被创建,当方法调用结束时,局部变量退栈,GC工作时回收对应内存。

  3. 匿名内部类对象生命周期与其它类对象一样:自创建一个匿名内部类对象,系统为该对象分配内存,直到没有引用指向分配给该对象的内存,它才有可能被GC回收。

  4. 所以完全可能出现的一种情况是:成员方法已调用结束,局部变量引用的对象已被GC回收,但匿名内部类的对象仍然活着。

  5. 如果匿名内部类的对象访问了局部变量,那么只要匿名内部类对象还活着,局部变量所引用对象就不能被GC回收。

  6. 解决方法:匿名内部类对象只能访问final类型的局部变量。定义为final后,编译器会把匿名内部类对象要访问的所有final类的型局部变量都拷贝一份作为它的成员变量。这样,即使局部变量已退栈,匿名内部类对象照样可以访问其引用的对象,因为它自己拷贝了一份作为成员变量,且与原局部变量引用的对象始终保持一致(final类型不可变)。

 

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

 

编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· [AI/GPT/综述] AI Agent的设计模式综述
点击右上角即可分享
微信分享提示