一般情况,我们把类定义成独立的单元。有些情况下,我们把一个类放在另一个类的内部定义,成为内部类(innerclasses)。
内部类可以使用public、default、 protected、private以及static修饰。而外部顶级类只能时候public和default修饰。
注意
内部类只是一个编译时概念,一旦我们编译成功,就会成为完全不同的两个类。对于一个名为Outer的外部类和其内部定义的名为Inner的内部类。编译完成后会出现Outer.class和Outer$Inner.class两个类的字节码文件。所以内部类是相对独立的一种存在,其成员变量/方法名可以和外部类相同。
内部类介绍
/**外部类*/
public class Outer{
private int age = 10;
public void show() {
System.out.println(age);//10
}
/**内部类Inner*/
public class Inner{
//内部类中可以声明与外部类同名的属性和方法
private int age = 20;
public void show() {
System.out.println(age); //20
}
}
}
编译后会产生两个不同的字节码文件,如图:
内部类的作用
- 内部类提供了更好的封装。只能让外部类直接访问,不允许同一个包中的其他类直接访问
- 内部类可以直接访问外部类的私有属性,内部类被当成其外部类的成员。但外部类不能访问内部类的内部属性。
- 接口只是解决了多重继承的部分问题,而内部类使得多重继承的解决方案变得更加完整。
内部类的使用场合
- 由于内部类提供了更好的封装性,并且可以很方便的访问外部类的属性。所以,在只为外部类提供服务的情况下可以优先考虑使用内部类。
- 使用内部类间接实现多继承:每个内部类都能独立地继承一个类或者实现某些接口,所以无论外部类是否已经继承了某个类或者实现了某些接口,对于内部类没有任何影响。
内部类的分类
在Java中内部类主要分为成员内部类(非静态内部类、静态内部类)、匿名内部类、局部内部类。
-
成员内部类(可以使用private、default、protected、public任意进行修饰。类文件:外部类$内部类.class)
-
非静态内部类(外部类里使用非静态内部类和平时使用其他类没什么不同)
成员变量的访问要点
public class Outer {
private int age = 10;
class Inner{
int age = 20;
public void show() {
int age = 30;
System.out.println("内部类方法里的局部变量age:" + age ); //30
System.out.println("内部类的成员变量age:" + this.age); //20
System.out.println("外部类的成员变量age:" + Outer.this.age); //10
}
}
}
new Inner()
- 外部类以外的地方使用非静态内部类:
Outer.Inner varname = new Outer().new Inner()
内部类的访问
public class TestInnerClass {
public static void main(String[] args) {
//先创建外部类实例,然后使用该外部类实例创建内部类实例
Outer.Inner inner = new Outer().new Inner();
inner.show();
Outer outer = new Outer();
Outer.Inner inn = outer.new Inner();
inn.show();
}
}
执行结果如图:
static class ClassName{
类体
}
- 当一个静态内部类对象存在,并不一定存在对应的外部类对象。因此,静态内部类的实例方法不能直接访问外部类的实例方法。
- 静态内部类看做外部类的一个静态成员。因此,外部类的方法中可以通过
静态内部类.名字
的方式访问静态内部类的静态成员,通过new 静态内部类()
访问静态内部类的实例。
静态内部类的访问
public class TestStaticInnerClass {
public static void main(String[] args) {
//通过new 外部类名.内部类名()来创建内部类对象
OuterClass.InnerClass innerClass = new OuterClass.InnerClass();
}
}
class OuterClass{
//相当于外部类的一个静态成员
static class InnerClass{
}
}
语法:
new 父类构造器(实参列表) \实现接口() {
//匿名内部类类体!
}
匿名内部类的使用
public class TestAnonymousInnerClass {
public static void test01(A a) {
a.a();
}
public static void main(String[] args) {
TestAnonymousInnerClass.test01(new A(){
@Override
public void a() {
System.out.println("TestAnonymousInnerClass.main(...).new AA(){...}.a");
}
});
}
}
interface A{
void a();
}
注意
- 匿名内部类没有访问修饰符。
- 匿名内部类没有构造方法。因为它连名字都没有。
-
局部内部类
还有一种内部类,它是定义在方法内部的,作用域只限于本方法,成为局部内部类。
局部内部类的使用主要是用来解决比较复杂的问题,想创建一个类来辅助我们的解决方案,到那时又不希望这个类是公共可用的,所以就产生了局部内部类。局部内部类和成员内部类一样被编译,只是它的作用域发生了改变,它只能在该方法中被使用,出了该方法就会失效。
局部内部类在实际开发中用的很少。
方法中的内部类
public class Test2 {
public void show() {
//作用域仅限于该方法
class Inner{
public void fun() {
System.out.println("helloworld");
}
}
new Inner().fun();
}
public static void main(String[] args) {
new Test2().show();
}
}