五、内部类
内部类(Inner Class)就是定义在一个类里面的类。与之对应,包含内部类的类被称为外部类。内部类可以用private修饰。
1、为什么要定义内部类?或者内部类的作用是什么?
内部类提供了更好的封装,可以把内部类隐藏在外部类之内,不允许同一个包中的其他类访问该类。
内部类的方法可以直接访问外部类的所有数据,包括私有的数据。
内部类所实现的功能使用外部类同样可以实现,只是有时使用内部类更方便。
2、内部类可分为几种?
成员内部类;静态内部类;方法内部类;匿名内部类。
3、成员内部类
内部类中最常见的就是成员内部类,也称为普通内部类。
代码示例:成员内部类
1 // 外部类Outer 2 public class Outer { 3 // 外部类的私有属性 4 private int a = 99; 5 private int b = 666; 6 7 // 内部类Inner 8 public class Inner { 9 // 内部类的成员属性 10 private int a = 199; 11 12 public void test() { 13 int a = 299; 14 System.out.println("访问方法的局部变量a:" + a); 15 System.out.println("访问内部类中的a:" + this.a); 16 System.out.println("访问外部类中的a:" + Outer.this.a); 17 System.out.println("不重名的变量b:" + b); 18 } 19 } 20 21 // 测试成员内部类 22 public static void main(String[] args) { 23 Outer o = new Outer(); 24 Inner i = o.new Inner(); 25 i.test(); 26 } 27 } 28 29 // 结果 30 访问方法的局部变量a:299 31 访问内部类中的a:199 32 访问外部类中的a:99 33 不重名的变量b:666
成员内部类的使用方法:
Inner类定义在Outer类的内部,相当于Outer类的一个成员变量的位置,Inner类可以使用任意访问控制符,如public、protected、private等。
Inner类中定义的test()方法可以直接访问Outer类中的数据,而不受访问控制符的影响,如直接访问Outer类中的私有属性a。
定义成员内部类后,必须使用外部类对象来创建内部类对象,而不能直接去new一个内部类对象,即:内部类 对象名 = 外部类对象.new 内部类( );
编译上面的程序后,会发现产生了两个.class 文件。
外部类是不能直接使用内部类的成员和方法。
如果外部类和内部类具有相同的成员变量或方法,内部类默认访问自己的成员变量或方法。如果要访问外部类的成员变量,可以使用this关键字。如:
4、静态内部类
静态内部类是static修饰的内部类。
代码示例:静态内部类
1 // 外部类 2 public class Outer { 3 private int a = 1; // 外部类的非静态变量 4 private static int b = 2; // 外部类的静态变量,不同名 5 private static int c = 4; // 外部类的静态变量,同名 6 7 public static class Inner { 8 int c = 3; 9 10 public void test() { 11 System.out.println(new Outer().a); 12 System.out.println(b); 13 System.out.println(c); 14 System.out.println(Outer.c); 15 } 16 } 17 18 public static void main(String[] args) { 19 new Inner().test(); 20 } 21 } 22 23 // 结果 24 1 2 3 4
这种内部类的特点是:
静态内部类不能直接访问外部类的非静态成员,但可以通过 new 外部类().成员 的方式访问。
如果外部类的静态成员与内部类的成员名称相同,可通过"类名.静态成员"访问外部类的静态成员;如果外部类的静态成员与内部类的成员名称不相同,则可通过"成员名"直接调用外部类的静态成员。
创建静态内部类的对象时,不需要外部类的对象,可以直接创建:内部类 对象名 = new 内部类();
5、方法内部类
方法内部类就是内部类定义在外部类的方法中,方法内部类只在该方法的内部可见,即只在该方法内可以使用。
代码示例:静态内部类
1 // 外部类 2 public class MOuter { 3 // 外部类中的方法 4 public void show() { 5 final int a = 25; // 常量 6 int b = 13; // 变量 7 // 方法内部类 8 class MInner { 9 int c = 2; // 内部类中的变量 10 11 public void print() { 12 System.out.println("访问外部类的方法中的常量a:" + a); 13 System.out.println("访问内部类中的变量c:" + c); 14 } 15 } 16 MInner mi = new MInner(); // 创建方法内部类对象 17 mi.print(); 18 } 19 20 public static void main(String[] args) { 21 MOuter mo = new MOuter(); 22 mo.show(); 23 } 24 } 25 26 // 结果 27 访问外部类的方法中的常量a:25 28 访问内部类中的变量c:2
注意:由于方法内部类不能在外部类的方法以外的地方使用,因此方法内部类不能使用访问控制符和 static 修饰符。方法内部类中的方法只能访问被 final 修饰的局部变量。(经测试,好像也能访问)
6、匿名内部类
匿名内部类就是一个实现了某个接口的类。而该类没有名字,在使用时直接 new 接口名。
1 // 外部类 2 public class Outer { 3 public Inner getInner(final int num, String str) { 4 // 匿名内部类 5 return new Inner() { 6 @Override 7 public int getNum() { 8 return num + 1; 9 } 10 }; 11 } 12 13 public static void main(String[] args) { 14 Outer outer = new Outer(); 15 final Inner inner = outer.getInner(12, "tom"); 16 17 System.out.println(inner.getNum()); 18 } 19 20 } 21 22 interface Inner { 23 int getNum(); 24 } 25 26 // 结果 27 13
注意:匿名内部类是没有访问修饰符的。new 匿名内部类,这个类首先是要存在的。如果我们将那个Inner接口注释掉,就会出现编译出错。
注意getInner()方法的形参,第一个形参是用final修饰的,而第二个却没有。同时我们也发现第二个形参在匿名内部类中没有使用过,所以当所在方法的形参需要被匿名内部类使用,那么这个形参就必须为final。
匿名内部类是没有构造方法的。因为它连名字都没有何来构造方法。