目录

总结

嵌套类分类

静态内部类

重要的结论。

静态内部类中能声明哪些类,变量和方法?

继承方面

内部类

细分类

内部类中能声明哪些类,变量和方法?

内部类能访问外围类的哪些变量和方法?

内部类是怎样绑定外围对象?

继承方面

本地内部类

嵌套接口

总结

嵌套类.png

嵌套类分类

静态内部类(静态嵌套类/静态成员类/静态类)

内部类(非静态嵌套类)

内部成员类

本地内部类

匿名内部类

嵌套接口

静态内部类

重要的结论:如果一个类被声明为static(即static修饰class),只有一种情况,该类是静态内部类。

1. 静态内部类中能声明哪些类,变量和方法?

没有限制,可以声明各种类型的类,变量,方法和静态代码块,细分为:

类:

枚举类

静态内部类

内部类

接口

变量:

静态变量

实例变量

方法:

静态方法

实例方法

静态代码块

2. 静态内部类能访问外围类的哪些变量和方法?

静态内部类可以访问外围类的任何成员,包括外围类中声明为private的成员,分为:

外围类的静态变量和方法(含私有):直接访问

外围类的实例变量和方法(含私有):通过外围类的实例对象进行访问

静态内部类类似类的静态变量,不需要依赖外围类的实例对象而存在,可以看作是顶层类,可以直接通过外围类来访问。

3. 继承方面

在继承方面,静态内部类与外围类没什么分别,在访问权限允许的情况下:任何类都可以继承静态内部类,静态内部类也可以继承任何类(类没有声明为final)或实现任何接口。

public class OuterClass {
// 静态变量
final static boolean FLAG_VALUE = true;
private static String name = "Outer Class";
// 实例变量
private int age;
// 静态方法
private static String getName() {
return name;
}
// 实例方法
private void setAge(int age) {
this.age = age;
}
// 静态内部类
public static class StaticInner {
// 声明静态变量
final static int x = 1;
static int y = 2;
// 声明实例变量
int a;
// 声明静态代码块
static { }
// 声明枚举类
enum InnerEnum { }
// 声明静态内部类
static class Inner2 { }
// 声明接口
interface Inner3 { }
// 声明内部类
class Inner4 { }
// 声明静态方法
static void OperateStatic() {
System.out.println("静态内部类直接访问外围类的静态变量:name = " + name + ", FLAG_VALUE = " + FLAG_VALUE);
System.out.println("静态内部类直接访问外围类的静态方法:getName : " + getName());
// 静态内部类不能直接访问外围类的实例变量和方法
//age = 30;
//setAge(30);
// 通过声明外围类对象,访问外围类对象的实例变量和方法
OuterClass outerClass = new OuterClass();
outerClass.age = 30;
outerClass.setAge(30);
}
// 声明实例方法
void operate() {
System.out.println("静态内部类直接访问外围类的静态变量:name = " + name + ", FLAG_VALUE = " + FLAG_VALUE);
System.out.println("静态内部类直接访问外围类的静态方法:getName : " + getName());
// 静态内部类不能直接访问外围类的实例变量和方法
//age = 30;
//setAge(30);
// 通过声明外围类对象,访问外围类对象的实例变量和方法
OuterClass outerClass = new OuterClass();
outerClass.age = 30;
outerClass.setAge(30);
}
}
}

内部类

1. 细分类

内部成员类

本地内部类(本地类/局部类)

匿名内部类(匿名类)

2. 内部类中能声明哪些类,变量和方法?

内部类可以声明实例变量,实例方法,final类型的静态变量。

内部类不可以声明静态成员:包括静态变量,静态方法,静态内部类,嵌套接口,静态初始化块。

细分为:

类:

只能声明内部类

不能声明枚举类,静态内部类,接口

变量:

只能声明实例变量,final类型静态变量

不能声明静态变量

方法:

只能声明实例方法

不能声明静态方法

不能声明静态代码块

3. 内部类能访问外围类的哪些变量和方法?

没有限制,外围类的所有变量和方法(含私有)都可以直接访问。

public class OuterClass {
// 静态变量
final static boolean FLAG_VALUE = true;
private static String name = "Outer Class";
// 实例变量
private int age;
// 静态方法
private static String getName() {
return name;
}
// 实例方法
private int getAge() {
return age;
}
// 内部类
public class Inner {
// 内部类不能声明静态变量
//private static String innerName = "Inner Class";
// 内部类只能声明实例变量和final类型静态变量
private int a;
final static int x = 1;
// 内部类不能声明静态代码块,枚举类,静态内部类,接口(枚举类型和接口类型总是静态的)
//static { }
//enum InnerEnum { }
//static class Inner2 { }
//interface Inner3 { }
// 内部类声明内部类
class Inner4 { }
// 内部类不能声明静态方法
//static void OperateStatic() { }
// 内部类只能声明实例方法
void operate() {
System.out.println("内部类访问外围类的静态变量:name = " + name);
System.out.println("内部类访问外围类的静态final变量:FLAG_VALUE = " + FLAG_VALUE);
System.out.println("内部类访问外围类的实例变量:age = " + (age = 40));
System.out.println("内部类访问外围类的静态方法:getName() = " + getName());
System.out.println("内部类访问外围类的实例方法:getAge() = " + getAge());
}
}
}

4. 内部类是怎样绑定外围对象?

总结: 创建内部类对象(调用内部类的构造器)时,编译器会隐式地在内部类中声明一个final的外围类类型的成员变量,然后将外围类的对象,通过内部类的构造器传递给该final成员变量,用来将内部类对象绑定到外围类对象。

public class Outer {
public class Inner {
// 编译器自动隐式生成的外围类类型的成员变量
final Outer this$0;
// 通过内部类构造器将外围类的对象传递给this$0成员变量,实现内部类与外围类对象的绑定
public Inner(Outer outer) {
this$0 = outer;
super();
}
}
}

创建内部类对象,如下:

Outer outer = new Outer();

Outer.Inner inner = outer.new Inner();

创建内部类对象时,系统会自动将外围类的对象(outer)作为参数传入内部类的构造器中,可认为是下面的方式:

Outer.Inner inner = outer.new Inner(outer);

5. 继承方面

在访问权限允许的情况下:内部类可以继承任何类,也可以由任何类继承。

问:那内部类与静态内部类在继承方面有什么区别呢?

答:内部类的对象总是要依赖于外部对象,因此一个类A继承了一个内部类,则类A也必须要与内部类的外围类对象相绑定,否则产生编译错误。

1.反例:产生编译错误

public class A extends Outer.Inner { }
class Outer {
class Inner { }
}

2.错误原因:想要创建A类对象,即A a = new A(); 会调用A类构造器,A类继承内部类(Outer.Inner),则会调用内部类的构造器,

这时没有有效的外围类对象,则无法实现内部类对象与外围类对象的绑定,产生编译错误。

3.修正

方法一:在A类构造器中传递一个外围类的引用,通过外围类对象来调用内部类的构造器,

就相当于将外围类的对象传递给了内部类的构造器,实现了内部类对象与外围类对象的绑定。

public class A extends Outer.Inner {
public A(Outer outer) {
outer.super();
}
}
class Outer {
class Inner { }
}

方法二:外部类继承外部类,内部类继承内部类

public class A extends Outer {
class InnerA extends Outer.Inner { }
}
class Outer {
class Inner { }
}

创建A类的内部类InnerA对象:

A a = new A();

A.InnerA innerA = a.new InnerA();

创建内部类A.InnerA对象时,需要绑定外围对象,a引用就是外围类的对象,

内部类A.InnerA继承了另外一个内部类Outer.Inner,在A.InnerA调用父类构造器时,也需要传递父类的外围类(Outer)对象,

A类继承了Outer类,因为子类的对像可以当作父类的对象来使用,因此a引用也是另一个内部类Outer.Inner的外围对象。

6. 本地内部类

本地内部类:就是在方法,构造器,初始化块中声明的类。

本地内部类不是类的成员,从结构上类似一个局部变量,因此不能使用访问修饰符(public,protected,private),也不能使用static修饰。

public class LocalInnerDemo {
private int x = 100;
// 1.本地内部类声明在实例初始化块中
{
class Local1 { }
}
// 2.本地内部类声明在静态初始化块中
static {
class Local2 { }
}
// 3.本地内部类声明在构造器中
public LocalInnerDemo() {
int y = 2;
final int z = 3;
class Local3 {
int a = x;
int b = y;
int c = z;
}
}
// 4.本地内部类声明在实例方法中
public T1 method1() {
// 使用本地内部类实现某个接口,然后以接口形式返回
class Local4 implements T1 {
@Override
public void operate() {
System.out.println("Start to operate.");
}
}
return new Local4();
}
// 5.本地内部类声明在静态方法中
public static T1 method2() {
class Local5 implements T1 {
@Override
public void operate() {
System.out.println("Start to operate.");
}
}
return new Local5();
}
}
interface T1 {
void operate();
}

问:本地内部类声明在实例环境(实例方法,构造器,实例初始化块)和静态环境中有什么区别呢?

答:实例环境:本地内部类需要与外围类绑定,即会在类中隐式生成一个final的引用。

静态环境:本地内部类不需要与外围类绑定。

嵌套接口

嵌套接口:就是在类或者接口中声明的接口。

不管声明在类中,还是接口中,嵌套接口永远都是静态的。

当类实现了某个接口时,不需要实现嵌套接口的方法。