一.概念
当一个事物内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么将这个内部的完整结构最好使用内部类。在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();
}
}
注意:通过匿名内部类访问局部变量,此局部变量不能被修改