温习一下java的内部类
常规内部类
常规内部类没有用static修饰且定义在在外部类类体中
- 常规内部类中的方法可以直接使用外部类的实例变量和实例方法
- 在常规内部类中可以直接用内部类创建对象
/**
* @author durant
*/
public class MyOuter {
private int x = 100;
/**
* 创建一个内部类
*/
class MyInner {
private String y = "Hello!";
public void innerMethod() {
System.out.println("内部类中 String =" + y);
// 直接访问外部类中的实例变量x
System.out.println("外部类中的x =" + x);
outerMethod();
System.out.println("x is" + MyOuter.this.x);
}
}
public void outerMethod() {
x++;
}
public MyInner makeInner() {
return new MyInner();
}
public static void main(String[] args) {
// 使用外部类构造方法创建mo对象
MyOuter mo = new MyOuter();
// 常规内部类需要通过外部类的实例才能创建对象,与实例变量需要通过对象来访问相似
// 创建inner对象
MyOuter.MyInner inner = mo.new MyInner();
inner.innerMethod();
//可以调用外部类的创建内部类方法来得到内部类的对象.
MyOuter.MyInner inner1 = mo.makeInner();
inner1.innerMethod();
}
}
运行结果:
内部类中 String =Hello!
外部类中的x =100
x is101
内部类中 String =Hello!
外部类中的x =101
x is102
静态内部类
与类的其他成员相似,可以用static修饰内部类,这样的类称为静态内部类
- 静态内部类与静态内部方法相似,只能访问外部类的static成员,不能直接访问外部类的实例变量,与实例方法,只有通过对象引用才能访问
- 由于static内部类不具有任何对外部类实例的引用,因此static内部类中不能使用this关键字来访问外部类中的实例成员,但是可以访问外部类中的static成员。这与一般类的static方法相通
/**
* @author durant
*/
public class MyOuter {
public static int x=100;
public int a=3;
public static class MyInner{
private String y="Hello!";
public void innerMethod(){
System.out.println("x="+x);
System.out.println("y="+y);
// 不能访问到外部类中非静态的成员
//System.out.println(a);
}
}
public static void main(String[] args) {
// 静态内部类不通过外部实例就可以创建对象;与类变量可以通过类名访问相似
MyOuter.MyInner si=new MyOuter.MyInner();
MyInner myInner = new MyInner();
si.innerMethod();
myInner.innerMethod();
}
}
运行结果:
x=100
y=Hello!
x=100
y=Hello!
局部内部类
在方法体或语句块(包括方法、构造方法、局部块或静态初始化块)内部定义的类成为局部内部类
- 局部内部类不能加任何访问修饰符,因为它只对局部块有效
- 局部内部类只在方法体中有效,就想定义的局部变量一样,在定义的方法体外不能创建局部内部类的对象
在方法内部定义类时,应注意以下问题:
- 方法定义局部内部类同方法定义局部变量一样,不能使用private、protected、public等访问修饰说明符修饰,也不能使用static修饰,但可以使用final和 abstract修饰
- 方法中的内部类可以访问外部类成员。对于方法的参数和局部变量,必须有final修饰才可以访问
- static方法中定义的内部类可以访问外部类定义的static成员
/**
* @author durant
*/
public class MyOuter {
private int size = 5;
private int y = 7;
public Object makeInner(final int localVar){
final int finalLocalVar = localVar;
/**
* 创建内部类,该类只在makeInner()方法有效,就像局部变量一样。在方法体外部不能创建MyInner类的对象
*/
class MyInner{
int y=4;
@Override
public String toString(){
return "OuterSize:" + size +
"\nfinalLocalVar:" + finalLocalVar + " " + "this.y=" + y;
}
}
return new MyInner();
}
public static void main(String[] args) {
// 创建MyOuter对象obj,并调用它的makeInner()方法,该方法返回一个
Object obj = new MyOuter().makeInner(47);
// 该方法返回一个MyInner类型的的对象obj,然后调用其同toString方法。
System.out.println(obj.toString());
}
}
输出结果:
OuterSize:5
finalLocalVar:47 this.y=4
匿名内部类
如果某个类的实例只是用一次,则可以将类的定义与类的创建,放到一起完成
声明和构造匿名内部类的一般格式如下:
new ClassOrInterfaceName(){
/*类体*/
}
- 匿名内部类可以继承一个类或实现一个接口,这里的ClassOrInterfaceName是匿名内部类所继承的类名或实现的接口名。但匿名内部类不能同时实现一个接口和继承一个类,也不能实现多个接口。如果实现了一个接口,该类是Object类的直接子类,匿名类继承一个类或实现一个接口,不需要extends和implements关键字
- 由于匿名内部类没有名称,所以类体中不能定义构造方法,由于不知道类名也不能使用关键字来创建该类的实例。实际上匿名内部类的定义、构造、和第一次使用都发生在同样一个地方。此外,上式是一个表达式,返回的是一个对象的引用,所以可以直接使用或将其复制给一个对象变量
例:
TypeName obj=new Name(){
/*此处为类体*/
}
也可以将构造的对象作为调用的参数。例:
someMethod(new Name(){
/*此处为类体*/
});
/**
* @author durant
*/
public class MyOuter {
private int size=5;
public Object makeInner(int localVar){
final int finalLocalVar=localVar;
return new Object(){
//使用匿名内部类
public String toString(){
return "OuterSize="+size+"\nfinalLocalVar="+finalLocalVar;
}
};
}
public static void main(String args[]){
Object obj=new MyOuter().makeInner(47);
System.out.println(obj.toString());
}
}
输出结果:
OuterSize=5
finalLocalVar=47