java —— 内部类

时间:2022-09-25 15:14:33

_

普通内部类

  静态内部类

  局部内部类

  匿名内部类

内部类

  内部类是定义在另一个类中的类,定义内部类会起到的作用有以下三点:

   1、内部类方法访问该类定义所在的作用域中的数据,包括私有的数据。

   2、内部类可以对同一个包中的其他类隐藏起来

  3、当想定义一个回调函数且不想写大量代码,使用匿名(anonymous)内部类会更加便捷。

  下面用简单的内部类例子说明内部类的使用方式。

class OuterClass {
private int outId = 0;
private String outName = "out";
public void outerShow() {
System.out.println("OuterClass");
System.out.println(outId + ":" + outName);
} public class InnerClass {
private String inName="inner";
public void InnerShow() {
System.out.println("InnerClass");
System.out.println(outName + ":"+ this.inName);
}
}
}
public class InnerClassDemo {
public static void main(String[] args) {
OuterClass out = new OuterClass();
OuterClass.InnerClass in = out. new InnerClass ();
in.InnerShow();
}
}

  通过代码可以发现,

   1、内部类也可以用访问修饰进行控制(private public protected)

  2、若内部类的是公开的 (public) ,可以直接在其它地方创建该类的实例。前提是该内部类的外部类的对象必须已经存在。代码如下:OuterClass.InnerClass in = out. new InnerClass ();

  3、内部类也可以使用 this 进行属性调用

  4、内部类既可以访问自己的数据域,也可以访问创建它的外围类对象的数据域。

  这是因为,内部类的对象总有一个隐式引用,指向它的外部类对象。如图:

java —— 内部类

  这个引用在内部类的定义中是不可见的,现在我们将外围类对象的引用称为 outer。于是 InnerShow 方法将等价干下列形式形式:

public void InnerShow() {

System.out.println("InnerClass");
System.out.println(outer.outName + ":"+ outer.inName);
}

  外部类的引用在构造器中设置。编译器修改了所有的内部类的构造器,添加一个外部类引用的参数。因为 InnerClass 类没有定义构造器,所以编译器为这个类生成了一个默认的构造器,其代码如下所示: 

public InnerClass(OuterClass out){

  outer=out 
}

  备注:outer 不是Java 的关键宇。只是用它说明内部类中的机制。

内部类的特殊语法规则

  在上面,已经讲述了内部类有一个外部类的引用 outer。事实上,使用外部类引用的正规语法还要复杂一些。

  表达式 :   OuterClass.this.  ——表示外部类引用。

  例如,可以像下面这样编写 InnerClass 内部类的 InnerShow 方法  

public void InnerShow() {
System.out.println("InnerClass");
System.out.println(OuterClass.this.outName + ":"+ OuterClass.this.inName);
}

   通常 this 限定词是多余的。不过,可以通过这种方式区分内部类与外部类属性名相同情况时,内部类的调用问题。

   重点:内部类中声明的静态属性必须是 final。原因很简单,我们希望一个静态域只有一个实例,不过对于每个外部对像,会分别有一个单独的内部类实例。如果这个不是 final ,它可能就不是唯一的。

  内部类不可以定义 static 方法

静态内部类

  关键字 static 中提到 static 可以修饰成员变量、方法、代码块,其他它还可以修饰内部类,使用static修饰的内部类我们称之为静态内部类,也称之为嵌套内部类

  静态内部类与非静态内部类之间存在一个最大的区别,我们知道非静态内部类在编译完成之后会隐含地保存着一个引用,该引用是指向创建它的外围内,但是静态内部类却没有。没有这个引用就意味着:

   1、 它的创建是不需要依赖于外围类的。

   2、 它不能使用任何外围类的非static成员变量和方法。

局部内部类

  局部类不能用 public 或 private 访问说明符进行声明,它的作用域被限定在声明这个局部类的块中。

  局部类有一个优势,即对外部世界可以完全地隐藏起来。即使 OuterClass 类中的其他代码也不能访问它。除 outShow 方法外没有任何方法知道 AnonymousClass 类的存在。 

public void outerShow() {
System.out.println("OuterClass");
System.out.println(outId + ":" + outName);
int age=123;
class AnonymousClass{
private void show(){
System.out.println(age);
}
}
new AnonymousClass().show();
}

匿名内部类

  匿名内部类的常用方式,实现函数的回调

public abstract class Bird {
private String name; public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public abstract int fly();
} public class Test { public void test(Bird bird){
System.out.println(bird.getName() + "能够飞 " + bird.fly() + "米");
} public static void main(String[] args) {
Test test = new Test();
test.test(new Bird() { public int fly() {
return 10000;
} public String getName() {
return "大雁"
;
}
});

}
}
------------------
Output:
大雁能够飞 10000米

  在使用匿名内部类的过程中,我们需要注意如下几点:   

  1、使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口

  2、匿名内部类中是不能定义构造函数的。

  3、匿名内部类中不能存在任何的静态成员变量和静态方法。

  4、匿名内部类为局部内部类,所以局部内部类的所有限制同样对匿名内部类生效。

  5、匿名内部类不能是抽象的,它必须要实现继承的类或者实现的接口的所有抽象方法。

匿名内部类初始化

  经常使用构造器来完成某个实例的初始化工作的,但是匿名内部类是没有构造器的!对此,我们可以使用构造代码块进行初始化工作。 演示代码如下: 

interface InnerClass{
public String getName(); public int getAge();
}
public class OuterClass {
//java 8之前需要 final int age,final String name
public InnerClass getInnerClass(int age,String name){
return new InnerClass() {
int age_ ;
String name_;
//构造代码块完成初始化工作
{
if(0 < age && age < 200){
age_ = age;
name_ = name;
} }
public String getName() {
return name_;
} public int getAge() {
return age_;
}
};
}
public static void main(String[] args) {
OuterClass out = new OuterClass();
InnerClass inner_1 = out.getInnerClass(201, "chenssy");
System.out.println(inner_1.getName());
InnerClass inner_2 = out.getInnerClass(23, "chenssy");
System.out.println(inner_2.getName());
}
}

  

  备注:在Java SE 8 之前,必须把局部类和匿名内部类访问的局部变量声明为 final 。