一.概念
当一个事物内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么将这个内部的完整结构最好使用内部类。在Java中,可以将一个类定义在另一个类或者一个方法内部,前者称为内部类,后者称为外部类,内部类也是封装的一种体现
注意:
1.定义在类名{}外部的,即使在同一个文件里,都不能称为内部类
2.内部类和外部类共用同一个java源文件,但是经过编译后,内部类会形成单独的字节码文件(一个类一个字节码文件)
二.内部类的分类
内部类可以在一个类的哪些位置进行定义呢?
public class Outclass{//成员位置定义,未被static修饰-实例内部类public class Innerclass1{}//成员位置定义,被static修饰-静态内部类static class Innerclass2{}public void method(){//方法中也可以定义内部类-局部内部类,几乎不用class Innerclass3{}}
}
根据内部类定义的位置不同,一般可分为几种形式
三.实例内部类
即未被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);//输出300//如果要访问外部类同名成员时,必须:外部类名称.this.同名成员名字OutClass.this.c = 400;System.out.println(OutClass.this.c);//输出400}}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 outClass = new OutClass();OutClass.InnerClass innerClass2 = outClass.new InnerClass();innerClass2.methodInner();
四.静态内部类
被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();}}public static void main(String[] args) {//静态内部类对象创建&成员访问OutClass.InnerClass innerClass = new OutClass.InnerClass();innerClass.methodInner();}
}
不能直接在静态内部类中直接访问外部类的非静态成员,可以拿外部类对象访问
OutClass outClass = new OutClass();
System.out.println(outClass.a);
五.局部内部类
定义在外部类的方法体或者{}中,该种内部类只能在其定义的位置使用,一般使用较少
public class OutClass {int a;public void methodd(){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; 编译失败}
}
编译器也有自己独立的字节码文件。命名格式:外部类名字$数字内部类名字.class
六.匿名内部类
定义:匿名内部类其实就是没有名称的内部类
说明:在调用包含有接口类型参数的方法时吗,通常为了简化代码,可以直接通过匿名内部类的形式传入一个接口类型参数,在匿名内部类中直接完成方法的实现
前提:必须是类或接口
格式:new 类名/接口名(){
重写抽象方法
}
public abstract class Animal {public abstract void eat();
}
public class Demo {public static void main(String[] args) {new Animal(){ //整体就等效于:是Animal父类的子类对象public void eat(){System.out.println("狗吃肉");}}}
}
此时只是创建了子类对象,没有调用方法
若要调用方法,有两种方式
public class Demo {public static void main(String[] args) {new Animal(){ //整体就等效于:是Animal父类的子类对象public void eat(){System.out.println("狗吃肉");}}.eat();}
}
public class Demo {public static void main(String[] args) {Animal a = new Animal() {public void eat(){System.out.println("狗吃肉");}};a.eat();}
}
注意:通过匿名内部类访问局部变量,此局部变量不能被修改