- Java的四种内部类如下:
常规内部类 静态内部类 局部内部类 匿名内部类。
设计内部类的目的:如果想让一个类继承多个接口(可以多继承接口)继承多个抽象类是做不到。但是在设计比较复杂的时候需要继承多个接口或者抽象类,总不能全部功能写在一个类当中,所以为了实现继承多个接口或者抽象类,就设计了内部类。
另一方面,如果是单继承,一个复杂的功能,只能引进一个类来帮助,这个类会写很长。如果实现多继承,会使每个类的代码变短。逻辑清晰。
内部类的作用
初次接触内部类时,可能会产生这样的疑问:为什么需要内部类? java内部类有什么好处?
我们知道Java中类一般只继承一个类,它的多重继承在我们没有学习内部类之前是用接口来实现的。但使用接口有时候有很多不方便的地方。比如我们实现一个接口就必须实现它里面的所有方法。而有了内部类就不一样了。它可以使我们的类继承多个具体类或抽象类,并且还可以避免修改接口而实现同一个类中两种同名方法的调用。
举一个简单的例子,如果你想实现一个接口,但是这个接口中的一个方法和你构想的这个类中的一个方法的名称,参数相同,你应该怎么办?这时候,你可以建一个内部类实现这个接口。由于内部类对外部类的所有内容都是可访问的,所以这样做可以完成所有你直接实现这个接口的功能。
不过你可能要质疑,更改一下方法的不就行了吗?
的确,以此作为设计内部类的理由,实在没有说服力。真正的原因是通过Java内部类来继承接口,可以很好地解决Java相对C++的一个弱势问题:没有多继承。实际上,C++的多继承设计起来很复杂,而Java通过内部类+接口的设计方案,可以很好地实现多继承的效果。
常规内部类:常规内部类没有用static 修饰且定义在在外部类类体中。
1.常规内部类中的方法可以直接使用外部类的实例变量和实例方法。
2.在常规内部类中可以直接使用内部类创建对象
public class Myouter {
private int x =100;
class MyInner{
private String y="hello world";
public void innerMethod(){
System.out.println("内部类中 String = " + y);
System.out.println("外部类中的x =" + x);
// 直接访问外部类中的实例变量x
outerMethod();
System.out.println("x is" + Myouter.this.x);
}
}
public void outerMethod() {
x++;
}
public void makeInner(){
//在在外部类方法中创建内部类实例
MyInner in = new MyInner();
}
public static void main(String[] args) {
Myouter mo = new Myouter();
// 使用外部类构造方法创建mo对象
Myouter.MyInner inner = mo.new MyInner();
//常规内部类需要通过外部类的实例才能创建对象,与实例变量需要通过对象来访问相似
//创建inner对象
inner.innerMethod();
}
}
静态内部类:
与类的其他成员相似,可以用static修饰内部类对象,这样的类称为静态内部类。静态内部类与静态内部方法相似,只能访问外部类的static成员,不能直接访问外部类的实例变量和实例方法,只有通过对象引用才能访问。由于static内部类不具有任何对外部类实例的引用,因此static内部类中不能使用this关键字来访问外部类中的实例成员,但是可以访问外部类中的static成员。这与一般类的static方法相通。
package BinaryTree;
public class staticOuter {
public static int x=100;
public static class MyInner{
private String y="Hello!";
//public Object staticOuter;
public void innerMethod(){
System.out.println("x="+x);
System.out.println("y="+y);
//通对象调用外部非静态方法
staticOuter st = new staticOuter();
st.OuterClass();
//用类名调用外部静态方法
staticOuter.Outer();
}
public void OuterClass() {
// TODO Auto-generated method stub
System.out.println("this is inner class.");
}
}
public void OuterClass(){
System.out.println("this is outer class.");
}
public static void Outer(){
System.out.println("this is static outer class");
}
//
public static void main(String[] args) {
//静态内部类不通过外部实例就可以创建对象;与类变量可以通过类名访问相似
staticOuter.MyInner si = new staticOuter.MyInner();
si.innerMethod();
si.OuterClass();
System.out.println("------ outer object----");
staticOuter st = new staticOuter();
st.OuterClass();
}
}
局部内部类
在方法体或语句块(包括方法、构造方法、局部块或静态初始化块)内部定义的类成为局部内部类。
局部内部类不能加任何访问修饰符,因为它只对局部块有效。局部内部类只在方法体中有效,就像定义的局部变量一样,在定义的方法体外不能创建局部内部类的对象
在方法内部定义类时,应注意以下问题:
1.方法定义局部内部类同方法定义局部变量一样,不能使用private、protected、public等访问修饰
说明符修饰,也不能使用static修饰,但可以使用final和 abstract修饰
2.方法中的内部类可以访问外部类成员。对于方法的参数和局部变量,必须有final修饰才可以访问。
3.static方法中定义的内部类可以访问外部类定义的static成员
package BinaryTree;
public class JubuClass {
private int size=5,y=7;
public Object makeInner(int localVar){
final int finalLocalVar = localVar;
//创建内部类,该类只在makeInner()方法有效,就像局部变量一样。
//在方法体外部不能创建MyInner类的对象
class MyInner{
int y= 4;
public String toString(){
return "OuterSize"+size+ "\nfinalLocalVar"+" "+"this.y="+this.y;
}
}
return new MyInner();
}
}
package BinaryTree;
public class JubuClassMain {
public static void main(String[] args) {
Object obj = new JubuClass().makeInner(47);
System.out.println(obj.toString());
}
}
匿名内部类
定义类的最终目的是创建一个类的实例,但是如果某个类的实例只是用一次,则可以将类的定义与类的创建,放到与一起完成,或者说在定义类的同时就创建一个类以这种方法定义的没有名字的类成为匿名内部类。
声明和构造匿名内部类的一般格式如下:
new ClassOrInterfaceName(){
/*类体*/ }
package BinaryTree;
public class Niming {
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) {
// TODO Auto-generated method stub
Object obj = new Niming().makeInner(49);
System.out.println(obj.toString());
}
}
匿名内部类可以继承一个类或实现一个接口,这里的ClassOrInterfaceName是匿名内部类所继承的类名或实现的接口名。但匿名内部类不能同时实现一个接口和继承一个类,也不能实现多个接口。如果实现了一个接口,该类是Object类的直接子类,匿名类继承一个类或实现一个接口,不需要extends和implements关键字。
由于匿名内部类没有名称,所以类体中不能定义构造方法,由于不知道类名也不能使用关键字来创建该类的实例。实际上匿名内部类的定义、构造、和第一次使用都发生在同样一个地方。此外,上式是一个表达式,返回的是一个对象的引用,所以可以直接使用或将其复制给一个对象变量。