1、定义:在一个外部类中定义的类称为内部类。可以是直接在类中定义的,也可以是在方法中或者是代码块中定义的
2、内部类的分类:
内部类分为: 有名内部类和匿名内部类 静态内部类和非静态内部类
以下详述有名内部类和匿名内部类
3、有名内部类和匿名内部类
有名内部类(有类名):
public class Body {
class Heart{
void jump() {
System.out.println("正在跳动......");
}
}
}
匿名内部类(无类名):
因为创建对象需要类名,但是又因为匿名部类没有类名,所以匿名内部类定义的时候和创建对象一起定义,{}为类体。
(注意:匿名内部类没有class,大括号后要加 “ ; ”)
public class Body {
class Heart{//有名内部类
void jump() {
System.out.println("正在跳动......");
}
}
Object object = new Object (){//匿名内部类
};
}
4、语法
匿名内部类:
(1)、匿名内部类是普通类的子类
public static void main(String[] args) {
Mammal mammal = new Mammal() {
@Override
public void move() {
System.out.println("靠鳍游动......");
}
};
mammal.move();
}
Mammal mammal = new Mammal() {...匿名内部类...}不出错说明能调用Mammal里的方法,而且能重写Mammal里的方法,说明这个匿名内部类是Mammal普通类的子类。类似于上转型
调用move方法时,用到了多态,表面上调用的是父类方法,实际调用的是重写后的方法
输出结果:靠鱼鳍游动
(2)匿名内部类也可以是抽象类的子类
从上面运行结果(多态)可以看出,Mammal父类可以是抽象类
可以看到没有出错
(3)匿名内部类也可以是接口的实现类
可以看到同样没有出错
注:匿名内部类不能是抽象类,因为匿名内部类没有名字。如果可以是抽象类,那它必须有子类,而匿名内部类没有名字,也就不能创建子类
内部类:
首先要说明如果直接在类中定义内部类,则其使用方式与成员变量类似:
public class Body {
class Heart{
void jump() {
System.out.println("正在跳动......");
}
}
public static void main(String[] args) {
Mammal mammal = new Mammal() {
@Override
public void move() {
System.out.println("靠鳍游动......");
}
};
mammal.move();
Body body = new Body();
Heart heart = body.new Heart();
heart.jump();
}
}
访问权限用于限制成员变量或方法能否在另一个类中被使用,但因为局部变量已经限定,所以局部变量不能使用访问权限,如
1、外部类或外部接口 public和默认 的;
但是直接在类中定义的内部类(相当于成员变量),可以有public ~private
(当然是有名内部类,匿名的连名都没有,当然不会有访问控制符了)
2、在代码块中定义的内部类(局部变量),只在代码块中有效,且不能有访问控制符public~private
3、private修饰的成员变量在本类中有效,而内部类也属于本类,所以private修饰的成员变量可以在内部类中使用
总结:内部类可以直接定义在类中,也可以定义在方法或代码块中;其中直接定义在外部类中的内部类可以有public、protected、默认的和private四种访问权限修饰(普通外部类、接口和抽象类只能使用public和default修饰),也可以使用static修饰( static不能修饰普通外部类、接口和抽象类);但是定义在方法或代码块中的内部类不能有访问修饰符修饰,也不能有static修饰。
4、如果在内部类中使用局部变量的值,则:JDK8.0及以上版本可以不用final修饰,8.0以前必须使用final
JDK8.0(JavaSE-1.8)
javaSE-1.7: 不用final会报错
正确的应该加final
拓展补充:无论是匿名内部类还是有名内部类,javac(编译)都会产生一个独立的class文件:编译之后内部类会被编译成独立的.class文件,如果该内部类为有名内部类,则有名内部类字节码文件名为外部类的类名+$+内部类类名;如果为匿名内部类,则匿名内部类字节码文件名为外部类类名+$+数字;