Java内部类
一、 含义
在Java编程语言里,程序是由类(class)构建而成的。在一个类的内部也可以声明类,我们把这样的类叫做内部类。
二、 作用
•实现了更好的封装,我们知道,普通类(非内部类)的访问修饰符不能为private或protected,而内部类可以。当我们将内部类声明为private时,只有外部类可以访问内部类,很好地隐藏了内部类。
•内部类可以继承(extends)或实现(implements)其他的类或接口,而不受外部类的影响。
•内部类可以直接访问外部类的字段和方法,即使是用private修饰的,相反的,外部类不能直接访问内部类的成员。
三、 原理
内部类是一个编译时的概念,编译后会生成两个独立的class文件,如下:
public class Outer{ private String outerName = "outer"; class Inner{ private String innerName = "inner"; } }
编译后的文件如下图:
编译后Outer.Inner被重命名为Outer$Inner,句点(.)被替换成了美元符号($)。
四、 分类
Java内部类可分为成员内部类、局部内部类、匿名内部类、静态内部类。
1) 成员内部类
成员内部类可以看成是外部类的一个成员,在成员内部类中无法声明静态成员,但static final字段是个例外。我们知道加载类时,会先初始化静态成员,如果成员内部类有静态成员,那么内部类就会在外部类之前生成,而内部类是为外部类服务的,内部类在外部类之前就生成可能会脱离掌控。在实例化成员内部类时,成员内部类会持有一个外部类当前对象的引用,这样在成员内部类中就可以直接访问外部类的成员,即使是private修饰的。
import static java.lang.System.out; public class Outer{ private String outerName = "outer"; //外部类无法直接访问内部类的成员,需要实例化内部类对象 private Inner inner = new Inner(); public class Inner{ private String innerName = "inner"; public void show(){ out.println(outerName); //可以直接访问外部类的成员 } } public void show(){ out.println(inner.innerName); inner.show(); } public static void main(String[] args){ Outer outer = new Outer(); outer.show(); //实例化内部类 Outer.Inner inner = outer.new Inner(); inner.show(); } }
运行结果:
inner
outer
outer
成员内部类对外部类对象的引用,是通过在this前面加上外部类的名字构成的,这种形式叫作
限定-this,out.println(outerName)与out.println(Outer.this.outerName)是等价的。
2) 局部内部类
局部内部类的使用和成员内部类的使用基本一致,只是局部内部类定义在外部类的方法中,就像局部变量一样,并不是外部类的成员。局部内部类在方法外是无法访问到的,但它的实例可以从方法中返回,并且实例在不再被引用之前会一直存在。局部内部类也可以访问所在方法的局部变量、方法参数等,限制是局部变量或方法参数只有在声明为final时才能被访问。
import static java.lang.System.out; public class Outer{ private String outerName = "outer"; public void show(final String str){ //方法参数为final类型 class Inner{ public void print(){ out.println(outerName+str); } } Inner inner = new Inner(); inner.print(); } public static void main(String[] args){ Outer outer = new Outer(); outer.show(":lalala"); } }
运行结果:
outer:lalala
3) 匿名内部类
可以把匿名内部类想象成是没有类名的局部内部类,匿名内部类有以下特点:
1、匿名内部类不能有构造器,匿名内部类没有类名,肯定无法声明构造器。
2、匿名内部类必须继承或实现一个接口,指定给new的类型为匿名类的超类型,匿名类不能有显示的extends或implements子句,也不能有任何修饰符。
3、匿名内部类和成员内部类、局部内部类一样,也不能声明静态成员。
import static java.lang.System.out; public class Outer{ private String outerName = "outer"; public void show(final String str){ new Inner(){ //实现了Inner接口 public void print(){ out.println(outerName+str); } }.print(); } public static void main(String[] args){ Outer outer = new Outer(); outer.show(":lalala"); } } interface Inner{ void print(); }
运行结果:
outer:lalala
4) 静态内部类
静态内部类,有的书上也称为嵌套类,声明它时需要用static修饰符,静态内部类不同于前三种内部类,静态内部类不会持有外部类当前对象的引用,所以在静态内部类中无法访问外部类的非静态成员,可以这么说,静态内部类不依赖于外部类。
import static java.lang.System.out; public class Outer{ private String outerName = "outer"; private static int id = 123; private Inner inner = new Inner(); public static class Inner{ public void print1(){ //out.println(outerName); 无法访问外部类的非静态成员 out.println(id); } public static void print2(){ out.println(id); } } public void show(){ inner.print1(); } public static void main(String[] args){ Outer outer = new Outer(); outer.show(); Outer.Inner.print2(); //直接通过类名访问静态内部类 } }