内部类的作用
- 更好的封装,可以将一个类私有
- 内部类可以访问外部类所有的数据,包括私有的数据
- 内部类有时使用方便
根据定义的方式不同,内部类分为静态内部类,成员内部类,局部内部类,匿名内部类四种。
成员内部类 / 普通内部类
1、没有使用 static 修饰的内部类
2、在成员内部类中不允许出现静态变量和静态方法的声明(final 修饰的除外),这是因为成员内部类是非静态的,类初始化的时候先初始化静态成员,如果允许成员内部类定义静态变量,那么成员内部类的静态变量初始化顺序是有歧义的。
3、成员内部类可以访问外部类中所有的成员(变量、方法),包含私有成员
4、构建内部类的实例,要求必须外部类的实例先存在
注意:
1、 外部类是不能直接使用内部类的成员和方法滴,可以通过内部类的实例对象来访问。
2、 如果外部类和内部类具有相同的成员变量或方法,内部类默认访问自己的成员变量或方法,如果要访问外部类的成员变量,可以使用 this 关键字。例如:外部类.this.成员变量
如何创建成员内部类的实例?
-
在外部类的外部或外部类的静态方法中:new Outer().new Inner();
-
在外部类的实例方法:
new Inner(); 或 this.new Inner();
public class Test02 {
public static void main(String[] args) {
Outer.Inner mm = new Outer().new Inner();
mm.test3();
System.out.println("*********");
Outer mo = new Outer();
mo.setS1(null);
}
}
class Outer {
private String s1 = "Outer";
public void setS1(String s1) {
this.s1 = s1;
// new Outer().new Inner();
this.new Inner(); //此时Outer已经实例化完成,所以可以使用this
new Inner().test3();
}
private void outerMethod() {
System.out.println("It's Method of OuterClass");
}
public static void test2() {
// new Outer().new Inner();
/*this.new Inner();
* 此时Outer没有实例化完成,所以不可以使用this
* static是在Outer构造器前使用,所以此时this不能使用
*/
String c = Outer.Inner.s4;
}
class Inner {
String s3 = "Inner";
static final String s4 = "static final";
void test3() {
System.out.println("s3 = " + s3);
System.out.println("s1 = " + s1);
Outer.this.outerMethod(); // 内部类成员方法与外部类成员方法同名时,使用this调用外部类的方法
outerMethod(); // 内部类没有同名方法时执行外部类的方法
}
/**
* 内部类可以调用外部类的方法,如果内部类有同名方法必须使用"OuterClass.this.MethodName()"格式调用
* (其中OuterClass与MethodName换成实际外部类名及其方法;this为关键字,表示对外部类的引用);
* 若内部类无同名方法可以直接调用外部类的方法
*/
private void outerMethod() {
System.out.println("It's Method of InnerClass");
}
}
}
静态内部类
定义在类内部的静态类,就是静态内部类。
public class Out {
private static int a;
private int b;
public static class Inner {
public void print() {
System.out.println(a);
}
}
}
1、访问特点可以类比静态变量和静态方法
2、脱离外部类的实例独立创建
- 在外部类的外部构建内部类的实例 new Outer.Inner();
- 在外部类的内部构建内部类的实例 new Inner();
3、静态内部类可以直接访问外部类中所有的静态成员,包含私有
4、Java集合类HashMap内部就有一个静态内部类Entry。Entry是HashMap存放元素的抽象,HashMap 内部维护 Entry 数组用了存放元素,但是 Entry 对使用者是透明的。像这种和外部类关系密切的,且不依赖外部类实例的,都可以使用静态内部类。
方法内部类
在方法中定义的内部类。方法内内部类只在该方法的内部中可以使用。
public class Out {
private static int a;
private int b;
public void test(final int c) {
final int d = 1;
class Inner {
public void print() {
System.out.println(c);
}
}
}
}
由于方法内部类不能在外部类的方法以外的地方使用,因此方法内部类不能使用访问控制符和 static 修饰符。
局部内部类可以访问的外部类的成员根据所在方法体不同
- 如果在静态方法中:可以访问外部类中所有的静态成员,包含私有
- 如果在实例方法中:可以访问外部类中所有的成员,包含私有
- 局部内部类可以访问所在方法中定义的局部变量,但要求局部变量必须使用 final 修饰
- 注意,若使用 JDK1.8,方法中内部类的方法是可以直接访问外部类的方法的局部变量,并且不需要 final 修饰
匿名内部类
匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类或者实现一个接口。同时它也是没有 class 关键字,这是因为匿名内部类是直接使用 new 来生成一个对象的引用。
public abstract class Bird {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public abstract int fly();
}
public class Test {
public void test(Bird bird) {
System.out.println(bird.getName() + "能够飞 " + bird.fly() + "米");
}
public static void main(String[] args) {
Test test = new Test();
test.test(new Bird() {
public int fly() {
return 10000;
}
public String getName() {
return "大雁";
}
});
}
}