Java基础(八) Java修饰符及接口详解之(二) 内部类
总体来熟,Java内部类为了从维护性的考虑,使用的并不是特别多。但是在某些情况下,类仅仅被使用一次,通常使用的就是匿名内部类。
- 内部类
8. 内部类
在一个类的内部定义的类称为内部类。
内部类分为:成员内部类 静态内部类 局部内部类 匿名内部类。
顶层类(正常类)只能处于public和默认访问级别,而成员内部类可以处于public, protected, private和默认这4种访问级别;
8.1 静态内部类(例:StaticOutterClass.java)
是成员内部类的一种,用static修饰。静态内部类具有以下特点:
静态内部类:(相对应类中的一个静态变量)
- 静态内部类中访问不到外部类的非静态属性或者方法
- 静态内部类的对象不需要依赖于外部类的对象
内部类 变量名字 = new 内部类();
public class A {
public static class B {
private int v;
public void say() {
System.out.println("hello");
}
}
public static void main(String[] args) {
B b = new B();
}
}
- 静态内部类可以直接访问外部类的静态成员,如果访问外部类的实例成员,就必须通过外部类的实例去访问。
class A {
private int a1; // 实例变量a1
private static int a2; // 静态变量a2
public static class B {
int b1 = a1; // 编译错误,不能直接访问外部类A的实例变量a1
int b2 = a2; // 合法,可以直接访问外部类A的静态变量a2
int b3 = A.this.a1; // 不合法 静态内部类中不能访问外部对象的this
}
}
- 在静态内部类中可以定义静态成员和实例成员。
class A {
public static class B {
int v1; // 实例变量
static int v2; // 静态变量
public static class C {
static int v3; // 静态内部类
}
}
}
- 可以通过完整的类名直接访问静态内部类的静态成员。
class A {
public static class B {
int v1; // 实例变量
static int v2; // 静态变量
public static class C {
static int v3; // 静态内部类
int v4;
}
}
}
public class Tester {
public void test() {
A.B b = new A.B();
A.B.C c = new A.B.C();
b.v1 = 1;
v.v2 = 1;
A.B.v1 = 1; // 编译错误
A.B.v2 = 1; // 合法
A.B.C.v3 = 1; // 合法
}
}
8.2 成员内部类:(相当于类中的一个成员变量)
- 成员内部类中不能有static的声明属性或者方法;
- 成员内部类可以由public protected default private修饰;
成员内部类是依赖于外部类的对象而存在的;
外部类.内部类 var = new 外部类().内部类();
例子:InstanceOutterClass.java
- 在创建实例内部类的实例时,外部类的实例必须已经存在。
Outer.InnerTool tool = new Outer().new InnerTool();
等价于:
Outer outer = new Outer();
Outer.InnerTool tool = outer.new InnerTool();
以下代码会导致编译错误:
Outer.InnerTool tool = new Outer.InnerTool();
- 实例内部类的实例自动持有外部类的实例的引用。在内部类中, 可以直接访问外部类的所有成员,包括成员变量和成员方法。
public class A {
private int a1;
public int a1;
static int a1;
public A(int a1, int a2) {
this.a1 = a1;
this.a2 = a2;
}
protected int methodA() {
return a1 * a2;
}
class B {
int b1 = a1; // 直接访问private的a1
int b2 = a2; // 直接访问public的a2
int b3 = a3; // 直接访问static的a3
int b4 = A.this.a1; // 访问类A的当前实例中的a1
int b5 = methodA(); // 访问methodA()方法
}
public static void main(String args[]) {
A.B b = new A(1, 2).new B();
System.out.println("b.b1=" + b.b1); // 打印b.b1=1;
System.out.println("b.b2=" + b.b2); // 打印b.b2=2;
System.out.println("b.b3=" + b.b3); // 打印b.b3=0;
System.out.println("b.b4=" + b.b4); // 打印b.b4=3;
System.out.println("b.b5=" + b.b5); // 打印b.b5=2;
}
}
- 外部类实例与内部类实例之间是一对多的关系,一个内部类实例只会引用一个外部类实例,而一个外部类实例对应零个或多个内部类实例。在外部类中不能直接访问内部类的成员,必须通过内部类的实例去访问。
class A {
class B {
private int b1 = 1;
public int b2 = 2;
class C {
}
}
public void test() {
int v1 = b1; // invalid
int v2 = b2; // invalid
B.C c1 = new C(); // invalid
B b = new B(); // valid
int v3 = b.b1; // valid
int v4 = b.b2; // valid
B.C c2 = b.new C(); // valid
B.C c3 = new B().new C(); // valid
}
}
- 实例内部类中不能定义静态成员,而只能定义实例成员。
- 如果实例内部类B与外部类A包含同名的成员,那么在类B中, this.v表示类B的成员, A.this.v表示类A的成员。
8.3 局部内部类:(相当于一个方法中的局部变量)
- 局部内部类不能用public private等修饰符修饰
- 写在方法当中,而且只能在方法当中使用
- 可以访问外层类的普通成员变量和静态成员变量以及普通方法和静态方法,也可以访问该内部类所在方法当中的局部变量,但是这个局部变量必须是final修饰;
例子:LocalOutterClass.java
局部内部类只能在当前方法中使用。
class A {
B b = new B(); // 编译错误;
public void method() {
class B {
int v1;
int v2;
class C {
int v3;
}
}
B b = new B(); // 合法
B.C c = b.new C(); // 合法
}
}
- 局部内部类和实例内部类一样,不能包含静态成员。
class A {
public void method() {
class B {
static int v1; // 编译错误
int v2; // 合法
static class C { // 编译错误
int v3;
}
}
}
}
- 在局部内部类中定义的内部类也不能被public、protected和private这些访问控制修饰符修饰;
- 局部内部类和实例内部类一样,可以访问外部类的所有成员,此外,局部内部类还可以访问所在方法中的final类型的参数和变量。
8.4 匿名内部类:(和局部内部类很相似)
- 匿名内部类也是用的最多的内部类
- 可以写成成员变量的形式,也可以写在方法当中,一般写在方法当中较多
- 匿名内部类里可以访问外部类的普通属性和方法,已经静态属性和方法,如果要访问这个内部类所在方法中的局部变量,那么要求这个局部变量必须是final修饰的
- 匿名内部类里面没有构造函数,因为这个类没有名字,所以在其他地方不能用
- 例子:AnonymousOutterClass.java
public class Hello {
public void test() {
// 假如A是同包下的一个接口,有一个抽象方法go
A a = new A() {
public void go() {
System.out.println("gogogo");
}
};
}
}
8.5 几种内部类的区别
8.5.1 创建
声明的位置:
- 静态内部类:类的内部,方法的外部,用static关键字修饰;
- 实例内部类:类的内部,方法的外部,不用static关键字修饰;
- 局部内部类:方法的内部;
- 匿名内部类:既可以在类的内部,方法的外部,也可以在方法的内部;
实例化方式:
- 静态内部类:new Outer.Inner();
//在外部类外创建;new Inner(); //在外部类内内部类外创建 - 实例内部类:new Outer().new Inner(); //在外部类外创建;
this.new Inner(); //在外部类内内部类外创建 - 局部内部类:new Inner(); //只能在方法内部创建;
- 匿名内部类:new 类名() {};
- 静态内部类:new Outer.Inner();
8.5.2 访问
外部类访问内部类:
- 静态内部类:通过完整的类名直接访问静态内部类的静态成员;
- 实例内部类:通过内部类的实例去访问内部类的成员;
- 局部内部类:不能访问;
- 匿名内部类:不能访问;
内部类访问外部类:
- 静态内部类:直接访问外部类的静态成员;
实例内部类:可以直接访问外部类的所有成员;如果实例内部类B与外部类A包含同名的成员,那么在类B中, this.v表示类B的成员,A.this.v表示类A的成员。
局部内部类:可以直接访问外部类的所有成员, 访问所在方法中的final类型的参数和变量;
- 匿名内部类:可以直接访问外部类的所有成员, 访问所在方法中的final类型的参数和变量;
参考文献
[1] 为什么Java内部类要设计成静态和非静态两种?
https://www.zhihu.com/question/28197253
[2]深入浅析Java中Static Class及静态内部类和非静态内部类的不同
http://www.jb51.net/article/74838.htm