前言
当一个事物的内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么这个内部的完整结构最好使用内部类。在 Java 中,可以将一个类定义在另一个类或者一个方法的内部,前者称为内部类,后者称为外部类。内部类也是封装的一种体现。
内部类和外部类共用同一个java源文件,但是经过编译之后,内部类会形成单独的字节码文件。
一、实例内部类
未被static修饰的成员内部类。
代码示例如下:
public class OutClass {
private int a;
static int b;
int c;
public void methodA(){
a = 10;
System.out.println(a);
}
public static void methodB(){
System.out.println(b);
}
// 实例内部类:未被static修饰
class InnerClass{
int c;
public void methodInner(){
// 在实例内部类中可以直接访问外部类中:任意访问限定符修饰的成员
a = 100;
b =200;
methodA();
methodB();
// 如果外部类和实例内部类中具有相同名称成员时,优先访问的是内部类自己的
c = 300;
System.out.println(c);
// 如果要访问外部类同名成员时候,必须:外部类名称.this.同名成员名字
OutClass.this.c = 400;
System.out.println(OutClass.this.c);
}
}
public static void main(String[] args) {
// 外部类:对象创建 以及 成员访问
OutClass outClass = new OutClass();
System.out.println(outClass.a);
System.out.println(OutClass.b);
System.out.println(outClass.c);
outClass.methodA();
outClass.methodB();
// 要访问实例内部类中成员,必须要创建实例内部类的对象
// 而普通内部类定义与外部类成员定义位置相同,因此创建实例内部类对象时必须借助外部类
// 创建实例内部类对象
OutClass.InnerClass innerClass1 = new OutClass().new InnerClass();
// 上述语法比较怪异,也可以先将外部类对象先创建出来,然后再创建实例内部类对象
OutClass.InnerClass innerClass2 = outClass.new InnerClass();
innerClass2.methodInner();
}
注意事项:
外部类中的任何成员都可以在实例内部类方法中直接访问。
实例内部类所处的位置与外部类成员位置相同,因此也受public、private等访问限定符的约束。
在实例内部类方法中访问同名的成员时,优先访问自己的,如果要访问外部类同名的成员,必须外部类名称.this.同名成员来访问。
实例内部类对象必须在先有外部类对象前提下才能创建。
实例内部类的非静态方法中包含了一个指向外部类对象的引用。
外部类中,不能直接访问实例内部类中的成员,如果要访问必须先要创建内部类的对象。
二、静态内部类
被static修饰的内部成员类称为静态内部类。
代码示例如下:
public class OutClass {
private int a;
static int b;
public void methodA(){
a = 10;
System.out.println(a);
}
public static void methodB(){
System.out.println(b);
}
// 静态内部类:被static修饰的成员内部类
static class InnerClass{
public void methodInner(){
// 在内部类中只能访问外部类的静态成员
// a = 100; // 编译失败,因为a不是类成员变量
b =200;
// methodA(); // 编译失败,因为methodB()不是类成员方法
methodB();
}
}
public static void main(String[] args) {
// 静态内部类对象创建 & 成员访问
OutClass.InnerClass innerClass = new OutClass.InnerClass();
innerClass.methodInner();
}
}
注意事项:
在静态内部类中只能访问外部类中的静态成员。
创建静态内部类对象时,不需要先创建外部类对象。
三、局部内部类
定义在外部类的方法体或者{}中,该种内部类只能在其定义的位置使用,一般使用的非常少,此处简单了解下语法格式。
代码示例如下:
public class OutClass {
int a = 10;
public void method(){
int b = 10;
// 局部内部类:定义在方法体内部
// 不能被public、static等访问限定符修饰
class InnerClass{
public void methodInnerClass(){
System.out.println(a);
System.out.println(b);
}
}
// 只能在该方法体内部使用,其他位置都不能用
InnerClass innerClass = new InnerClass();
innerClass.methodInnerClass();
}
public static void main(String[] args) {
// OutClass.InnerClass innerClass = null; 编译失败
}
}
注意事项:
局部内部类只能在所定义的方法体内部使用。
不能被public、static等修饰符修饰。
编译器也有自己独立的字节码文件,命名格式:外部类名字$数字内部类名字.class。
几乎不会使用。
四、匿名内部类
代码示例如下:
interface InterFaceA {
void func();
}
public class Test {
public static void main2(String[] args) {
TestA testA = new TestA();
System.out.println("=======");
InterFaceA a = new InterFaceA() {
@Override
public void func() {
System.out.println("哈哈哈 不懂了吧!");
}
};
a.func();
System.out.println("=======");
new InterFaceA() {
@Override
public void func() {
System.out.println("哈哈哈 不懂了吧!");
}
}.func();
}
}
以上代码定义了一个接口,通过匿名内部类的用法可以对接口内方法直接进行重写并且使用。匿名内部类的用法也可以用在普通类上面,例如以下代码示例:
public class Test {
public void func() {
System.out.println("func()");
}
public static void main1(String[] args) {
Test test = new Test();
test.func();
test.func();
new Test().func();//匿名对象
new Test().func();//匿名对象
}
}