java内部类有什么用?
搬运一下oracle文档:
https://docs.oracle.com/javase/tutorial/java/javaOO/nested.html
总结起来就是三点:
1. 有些类只对另外一个类有用, 这样可以组合在一起,让整个包更精简。比如Builder常常作为静态内部类。
2. 增加封装性。 比如有两个类A和B,B想要access A的成员。 如果将B作为A的内部类,A的成员可以保持private,并且B本身也被封装起来了。
3. 增加可读性和维护性。让代码离它被用到的地方更近了。
内部类(inner class)
- 拥有外部类的引用,这样内部类可以访问创建它的外部类的内容,甚至包括私有变量。
- 一旦编译成功,就会成为完全不同的两类。
对于一个名为outer的外部类和inner的内部类,编译完之后出现outer.class和outer$inner.class两类。 - 如果在外部类外部访问内部类,用out.inner
在外部类外部要生成一个内部类对象:Outer.Inner in = Outer.new.Inner() - 外部类可以访问内部类的private成员,神奇不神奇。
编译器实现的时候是这样的:Outer类和Inner类不再是嵌套结构,而是变为一个包中的两个类,然后,对于private变量的访问,编译器会生成一个accessor函数....... - 非静态内部类不能有静态数据、静态方法、静态内部类。
为啥呢?因为,非静态内部类的访问必须通过外部类的实例,所以要访问非静态内部类的静态方法,必须先实例化内部类和外部类,这样就和静态成员的设计是违背的。
静态内部类(静态嵌套类)(nested class)
- 实际上是一种外部类,因为与外部类的实例无关
- 定义在类中,方法外
- 只能访问外部类的静态成员(可以访问私有的静态成员)
- new的时候不需要外部类成员:Outer.Inner in = new Outer.Inner();
局部内部类
在一个方法甚至一个代码块之内。
可以被final或abstract修饰
不能含有静态成员
局部内部类访问外部类方法的局部变量要求其必须为final。
局部变量是随着方法的调用而调用,随着调用完毕而消失。而堆内存的对象内容并不会立即消失。
或者简单的来说是作用域的问题。就好像方法外面做的事情并不能改变方法内才定义的变量,因为你并不知道方法里面这个时候已经存在了这个局部变量了没有。在这个内部类中方法里面的本地变量是失效的,也就是不在作用域内,所以是不能够访问的。但是为什么这里用final却又可以访问呢? 因为Java采用了一种copy local variable的方式来实现,也就是说把定义为final的局部变量拷贝过来用,而引用的也可以拿过来用,只是不能重新赋值。从而造成了可以access local variable的假象,而这个时候由于不能重新赋值,所以一般不会造成不可预料的事情发生。
则java编译器则会在内部类NewAge内生成一个外部变量的拷贝,而且可以既可以保证内部类可以引用外部属性,又能保证值的唯一性
也就是拷贝了一个变量的副本,提供给局部内部类,这个副本的生命周期和局部内部类一样长,并且这个副本不可以修改,保证了数据的同步
注意:在Java8 中,被局部内部类引用的局部变量,默认添加final,所以不需要添加final关键词
匿名内部类
- 匿名类(Anonymouse Class)是一种没有类名的内部类,多用在事件处理的程序中。有时用户需要定义一个类,且只想在程序中定义该类的一个对象,并把它作为参数传递给一个方法。只要该类是一个现有类的派生或实现一个接口,就可以使用匿名类。匿名类定义短而简单且使用方便,但不易过多使用,它会引起程序代码的复杂化而不易理解
- 没有名字,所以不能定义构造方法。