前言
本文主要介绍:Java中类的成员:内部类
类的内部成员: 内部类
引入:
当一个事物的内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么整个内部的完整结构最好使用内部类。在Java中,允许一个类的定义位于另一个类的内部,前者称为内部类,后者称为外部类。Inner class一般用在定义它的类或语句块之内,在外部引用它时必须给出完整的名称。Inner class的名字不能与包含它的外部类类名相同;
分类: 成员内部类(static成员内部类和非static成员内部类);局部内部类(不谈修饰符)、匿名内部类。
成员内部类作为类的成员的角色:
- 和外部类不同,Inner class还可以声明为private或protected;
- 可以调用外部类的结构
- Inner class 可以声明为static的,但此时就不能再使用外层类的非static的成员变量;
成员内部类作为类的角色:
- 可以在内部定义属性、方法、构造器等结构
- 可以声明为abstract类 ,因此可以被其它的内部类继承
- 可以声明为final的
- 编译以后生成OuterClass$InnerClass.class字节码文件(也适用于局部内部类)
注意:
- 非static的成员内部类中的成员不能声明为static的,只有在外部类或static的成员内部类中才可声明static成员。
- 外部类访问成员内部类的成员,需要“内部类.成员”或“内部类对象.成员”的方式
- 成员内部类可以直接使用外部类的所有成员,包括私有的数据
- 当想要在外部类的静态成员部分使用内部类时,可以考虑内部类声明为静态的
class Animal {
String name = "小明";
int age;
public void eat() {
System.out.println("进食");
}
// 静态成员内部类
static class Dog {
String name;
int age;
public void show() {
System.out.println("卡拉是条狗");
}
}
// 非静态成员内部类
class Bird {
String name = "杜鹃";
public Bird() {
}
public void sing() {
System.out.println("我是一只小小鸟");
Animal.this.eat();// 调用外部类的非静态属性
eat();
System.out.println(age);
}
public void display(String name) {
System.out.println(name);// 方法的形参
System.out.println(this.name);// 内部类的属性
System.out.println(Animal.this.name);// 外部类的属性
}
}
}
调用成员内部类
public class InnerClassTest {
public static void main(String[] args) {
//创建Dog实例(静态的成员内部类):
Animal.Dog dog = new Animal.Dog();
dog.show();
//创建Bird实例(非静态的成员内部类):
// Animal.Bird bird = new Animal.Bird();//错误的
Animal p = new Animal();
Animal.Bird bird = p.new Bird();
bird.sing();
System.out.println(“------------”);
bird.display("黄鹂");
}
}
局部内部类:
- 内部类仍然是一个独立的类,在编译之后内部类会被编译成独立的.class文件,但是前面冠以外部类的类名和$符号,以及数字编号。
- 只能在声明它的方法或代码块中使用,而且是先声明后使用。除此之外的任何地方都不能使用该类。
- 局部内部类可以使用外部类的成员,包括私有的。
- 局部内部类可以使用外部方法的局部变量,但是必须是final的。由局部内部类和局部变量的生命周期不同所致。
- 局部内部类和局部变量地位类似,不能使用public,protected,缺省,private
- 局部内部类不能使用static修饰,因此也不能包含静态成员
匿名内部类:
匿名内部类不能定义任何静态成员、方法和类,只能创建匿名内部类的一 个实例。一个匿名内部类一定是在new的后面,用其隐含实现一个接口或 实现一个类。
格式:
new 父类构造器(实参列表)|实现接口(){
//匿名内部类的类体部分
}
匿名内部类的特点:
- 匿名内部类必须继承父类或实现接口
- 匿名内部类只能有一个对象
- 匿名内部类对象只能使用多态形式引用
public class InnerClass {
// 开发中很少见
public void method(){
//局部内部类
class AA{
}
}
{
//局部内部类
class BB{
}
}
public Person(){
//局部内部类
class CC{
}
}
// 返回一个实现了Comparable接口的类的对象
public Comparable getComparable() {
// 创建一个实现了Comparable接口的类:局部内部类
// 方式一:
class MyComparable implements Comparable {
@Override
public int compareTo(Object o) {
return 0;
}
}
return new MyComparable();
// 方式二:匿名内部类
return new Comparable() {
@Override
public int compareTo(Object o) {
return 0;
}
};
}
}
/*
* 在局部内部类的方法中(比如:show)如果调用局部内部类所声明的方法(比如:method)中的局部变量(比如:num)的话,
* 要求此局部变量声明为final的。
*
* jdk 7及之前版本:要求此局部变量显式的声明为final的
* jdk 8及之后的版本:可以省略final的声明
*
*/
public void method() {
// 局部变量
int num = 10;
class AA {
public void show() {
// num = 20;//报错
System.out.println(num);
}
}
}