分类
Java中的内部类共分为四种:
静态内部类static inner class (also called nested class)
成员内部类member inner class
局部内部类local inner class
匿名内部类anonymous inner class
一.成员内部类:member inner class
成员内部类没有用static修饰且定义在外部类类体中。
成员内部类就像一个实例变量。需要通过外部类的实例才能创建对象,与实例变量需要通过对象来访问相似。
MyOuter.MyInner inner = new MyOuter.new MyInner();
它可以访问它的外部类的所有成员变量和方法,不管是静态的还是非静态的都可以。
成员内部类也是定义在另一个类中,但是定义时不用static修饰。
成员内部类和静态内部类可以类比为非静态的成员变量和静态的成员变量。
成员内部类就像一个实例变量。
它可以访问它的外部类的所有成员变量和方法,不管是静态的还是非静态的都可以。
在外部类里面创建成员内部类的实例:
this.new Innerclass();
在外部类之外创建内部类的实例:
new Outerclass().new Innerclass();
在内部类里访问外部类的成员:
Outerclass.this.member
示例:
public class Out {//外部类
private int age = 12;
class In {//内部类
private int age = 13;
public void print() {
int age = 14;
//内部类中 访问 内部类 成员变量
System.out.println("局部变量:" + age);//14
//内部类中 访问 内部类 实例变量
System.out.println("内部类变量:" + this.age);//13
//内部类中 访问 外部类 实例变量
System.out.println("外部类变量:" + Out.this.age);//12
}
}
}
public class Demo {//测试类
public static void main(String[] args) {
Out.In in = new Out().new In();
in.print();
}
}
输出:
局部变量:14
内部类变量:13
外部类变量:12
二、局部内部类:local inner class
局部内部类没有用static修饰且定义在在外部类的方法中。
在方法体或语句块(包括方法、构造方法、局部块或静态初始化块)内部定义的类成为局部内部类。
局部内部类不能加任何访问修饰符,因为它只对局部块有效。
局部内部类只在方法体中有效,就像定义的局部变量一样,在定义的方法体外不能创建局部内部类的对象
在方法内部定义类时,应注意以下问题:
1.方法定义局部内部类同方法定义局部变量一样,不能使用private、protected、public等访问修饰说明符修饰,也不能使用static修饰,但可以使用final和abstract修饰
2.方法中的内部类可以访问外部类成员。对于方法的参数和局部变量,必须有final修饰才可以访问。
3.static方法中定义的内部类可以访问外部类定义的static成员
public class Out {//外部类
private int age = 12;
public void Print(final int x) {//外部类的方法
class In {//内部类
public void inPrint() {//内部类的方法
System.out.println(x);
System.out.println(age);
}
}
new In().inPrint();
}
}
public class Demo {//测试类
public static void main(String[] args) {
Out out = new Out();
out.Print(3);
}
}
输出
3
12
三、静态内部类:static inner class (also called nested class)
与类的其他成员相似,可以用static修饰内部类,这样的类称为静态内部类。
静态内部类与静态内部方法相似,只能访问外部类的static成员,不能直接访问外部类的实例变量与实例方法,只有通过对象引用才能访问。
由于static内部类不具有任何对外部类实例的引用,因此static内部类中不能使用this关键字来访问外部类中的实例成员,但是可以访问外部类中的static成员。这与一般类的static方法相通。
被编译成一个完全独立的.class文件,名称为OuterClass$InnerClass.class的形式。
只可以访问外部类的静态成员和静态方法,包括了私有的静态成员和方法。
示例:
public class Out {//外部类
public static int x = 100;//静态
public static class MyInner {//内部类:静态
private String y = "Hello!";
public void innerMethod() {
System.out.println("x=" + x);
System.out.println("y=" + y);
}
}
}
public class Demo {//测试类
public static void main(String[] args) {
//静态内部类不通过外部实例就可以创建对象;与类变量可以通过类名访问相似
Out.MyInner si = new Out.MyInner();
si.innerMethod();
}
}
输出:
x=100
y=Hello!
四、匿名内部类:anonymous inner class
定义类的最终目的是创建一个类的实例,但是如果某个类的实例只是用一次,则可以将类的定义与类的创建,放到与一起完成,或者说在定义类的同时就创建一个类,以这种方法定义的没有名字的类成为匿名内部类。
声明和构造匿名内部类的一般格式如下:
new ClassOrInterfaceName(){
/*类体*/
};
1.匿名内部类可以继承一个类或实现一个接口,这里的ClassOrInterfaceName是匿名内部类所继承的类名或实现的接口名。但匿名内部类不能同时实现一个接口和继承一个类,也不能实现多个接口。如果实现了一个接口,该类是Object类的直接子类,匿名类继承一个类或实现一个接口,不需要extends和implements关键字。
2.由于匿名内部类没有名称,所以类体中不能定义构造方法,由于不知道类名也不能使用关键字来创建该类的实例。实际上匿名内部类的定义、构造、和第一次使用都发生在同样一个地方。此外,上式是一个表达式,返回的是一个对象的引用,所以可以直接使用或将其复制给一个对象变量。例:
TypeName obj=new Name(){
/*此处为类体*/
}
同样,也可以将构造的对象作为调用的参数。例:
someMethod(new Name(){
/*此处为类体*/
}
);
匿名内部类就是没有名字的局部内部类,不使用关键字class, extends, implements, 没有构造方法。
匿名内部类隐式地继承了一个父类或者实现了一个接口。
匿名内部类使用得比较多,通常是作为一个方法参数。
生成的.class文件中,匿名类会生成OuterClass$1.class文件,数字根据是第几个匿名类而类推。
示例:
public class Out {//外部类
private int size = 5;
public Object makeInner(int localVar) {
int finalLocalVar = localVar;
return new Object() {
//使用匿名内部类
public String toString() {
return "OuterSize=" + size + "\nfinalLocalVar=" + finalLocalVar;
}
};
}
/**
* @param args
*/
public static void main(String args[]) {
Object obj = new Out().makeInner(47);
System.out.println(obj.toString());
}
}
结果:
OuterSize=5
finalLocalVar=47