1、在 java 语言中,有一种类叫做内部类(inner class),也称为嵌入类(nested class),它是定义在其他类的内部。内部类作为其外部类的一个成员,与其他成员一样,可以直接访问其外部类的数据和方法。只不过相比较外部类只有 public 和默认的修饰符不同,内部类作为一个成员,可以被任意修饰符修饰。编译器在编译时,内部类的名称为OuterClass$InnerClass.class 。1、内部类访问数据变量当在某些时候,内部类中定义的变量与外部类中变量名称相同时,如何确保正确地访问每一个变量呢?1.1 在 main 中直接从外部类调用内部类的方法class Outerprivat
2、e int index = 10;class Inner private int index = 20;void print()int index = 30;System.out.println(this); / the object created from the InnerSystem.out.println(Outer.this); / the object created from the OuterSystem.out.println(index); / output is 30System.out.println(this.index); / output is 20System
3、.out.println(Outer.this.index); / output is 10void print() Inner inner = new Inner();/得到内部类的引用inner.print();class Test public static void main(String args) Outer outer = new Outer();outer.print();在这里内部类 Inner 中关键字 this 指向内部类 Inner 的对象,如果要想指向外部类的对象,必须在 this 指针前加上外部类名称,表示 this 是指向外部类构造的碎屑,如 Outer.this
4、 。1.2 在 main 中显式返回内部类引用class Outer private int index = 10;class Inner private int index = 20;void print() int index = 30;System.out.println(index);System.out.println(this.index);System.out.println(Outer.this.index);Inner getInner() return new Inner();/返回一个内部类的引用class Test public static void main(Str
5、ing args) Outer outer = new Outer();Outer.Inner inner = outer.getInner();inner.print();Inner 是 Outer 的内部类,所以在类 Test 中必须用属性引用符来标识出内部类。1.3 当 main 方法在 Outer 类内部class Outer private int index = 10;class Inner private int index = 20;void print() int index = 30;System.out.println(index);System.out.println(
6、this.index);System.out.println(Outer.this.index);Inner getInner() return new Inner();/返回一个内部类的引用public static void main(String args) Outer outer = new Outer();Inner inner = outer.getInner(); / 注意此处变化inner.print();因为 main 方法在 Outer 内部,故可以直接引用,不需要属性引用符。1.4 在 main 方法中直接产生内部类对象class Test public static v
7、oid main(String args) Outer outer = new Outer();Outer.Inner inner = outer.new Inner(); / 注意此处变化inner.print();在利用 new 构造方法构造一个外部类对象时,并没有连带着构造一个内部类对象,故需要访问内部类方法时,必须使用 new 操作符为这个外部类对象再构造一个内部类对象。2、局部内部类在方法中定义的内部类是局部内部类,它只能访问方法中的 final 类型的局部变量,因为用final 定义的局部变量相当于是一个常量,延长了其生命周期,使得方法在消亡时,其内部类仍可以访问该变量。另外,它同
8、样也可以引用定义在外部类的变量和方法。而且方法体中的局部内部类不允许有访问修饰符。class Outerint num=10; public void print(final int aArgs)class Innerint num=20;public Inner()System.out.println(“This is Inner.“);/此句可看出它与匿名内部类用法的不同。public void print() int num=30;System.out.println(this); / the object created from the local InnerSystem.out.p
9、rintln(num);System.out.println(this.num);System.out.println(Outer.this.num);System.out.println(aArgs);Inner inner=new Inner();/此句必须放在定义类 Inner 的后面inner.print();public static void main(String args)Outer outer=new Outer();outer.print(40);对于局部类的命名,不管是在一个方法中定义多个类还是在几个方法中分别定义类,其编译后命名是:OuterClass$1InnerCl
10、ass.class3、匿名内部类匿名内部类作为一种特殊的内部类,除了具有普通内部类的特点,还有自己的一些独有特性:匿名内部类必须扩展一个基类或实现一个接口,但是不能有显式的 extends 和 implements子句;匿名内部类必须实现父类以及接口中的所有抽象方法;匿名内部类总是使用父类的无参构造方法来创建实例。如果是实现了一个接口,则其构造方法是 Object();匿名内部类编译后的命名为:OuterClass$n.class ,其中 n 是一个从 1 开始的整数,如果在一个类中定义了多个匿名内部类,则按照他们的出现顺序从 1 开始排号。abstract class Aabstract p
11、ublic void sayHello();class Outerpublic static void main(String args)new Outer().callInner(new A()public void sayHello()System.out.println(this); / the object created from the anonymous InnerSystem.out.println(“Hello!“););public void callInner(A a)a.sayHello();4、静态内部类和非静态内部类相比,区别就在于静态内部类没有了指向外部类的引用。
12、除此之外,在任何非静态内部类中,都不能有静态数据,静态方法或者又一个静态内部类(内部类的嵌套可以不止一层) 。不过静态内部类中却可以拥有这一切。这也算是两者的第二个区别吧。一个静态的内部类,才可以声明一个 static 成员,静态内部类可以访问外围类的静态方法、成员(包括 private static 的成员) 。静态内部类实例化的时候不必先实例化外围类,可以直接实例化内部类。而对于非静态内部类则必须先实例化其外部类,才能再实例化本身。5内部类的继承当一个类继承自一个内部类时,缺省的构造器不可用。必须使用如下语法:class WithInnerclass Innerpublic void sa
13、yHello()System.out.println(“Hello.“);public class Test extends WithInner.InnerTest(WithInner wi)wi.super();public static void main(String args)WithInner wi=new WithInner();Test test=new Test(wi);test.sayHello(); 因为每一个内部类都有一个指向外部类的引用,在继承一个内部类,必须先创建一个外部类,通过这个外部类引用来调用其内部类的构造方法。如果继承的内部类是一个静态内部类,则就不需要这样,
14、直接 super()调用即可;6、内部类的 2 种特殊用法一个类从另一个类派生出来,又要实现一个接口。但在接口中定义的方法与父类中定义的方法的意义不同,则可以利用内部类来解决这个问题。interface Machinevoid run();class Person void run()System.out.println(“run“);class Robot extends Personprivate class MachineHeart implements Machinepublic void run()System.out.println(“heart run“);Machine get
15、Machine()return new MachineHeart();class Testpublic static void main(String args)Robot robot = new Robot();Machine m = robot.getMachine();m.run();robot.run();在 Robot 类内部使用内部类 MachineHeart 来实现接口 Machine 的 run 方法。同时 Robot 类又继承了父类 Person 的 run 方法。如果不使用内部类 MachineHeart 而使 Robot 直接实现接口 Machine,则该如何调用父类的
16、run 方法?利用内部类可解决 c+中多重继承所解决的问题class Avoid fn1()System.out.println(“It s fn1.“);abstract class Babstract void fn2();class C extends AB getB()return new B() public void fn2()System.out.println(“It s fn2.“);class Testpublic static void main(String args)C c = new C();c.fn1();c.getB().fn2();类 C 既要继承类 A 又要
17、继承类 B,则可将类 B 的定义放入类 C 内部,使之成为内部类。一般情况下 当我们需要在某一情形下实现一个接口,而在另一情形下又不需要实现这个接口时,我们可以使用内部类来解决这一问题。让内部类来实现这个接口。另外一个很好的理由是 java 内部类加上接口可以有效地实现多重继承。Java 中内部类的编写内部类的相关知识点:1) 在一个类中定义另外一个类,这个类就叫做内部类或内置类 (inner class) 。2) 内部类可以让我们将逻辑上相关的一组类组织起来,并由外部类(outer class)来控制内部类的可见性。内部类可以访问外部类的任何类型的方法和成员变量。3) 当我们建立一个 inn
18、er class 时,其对象就拥有了与外部类对象之间的一种关系,这是通过一个特殊的 this reference 形成的,使得内部类对象可以随意的访问外部类中所有的成员。下面为内部类的简单实现:class Outerprivate int index=100;void print() Inner i=new Inner();i.print();class Innervoid print()System.out.println(index);class Testpublic static void main(String args)Outer o=new Outer();o.print();Ou
19、ter.Inner inner=outer.getinner();inner.print();1) 在方法中定义的内部类,如果要访问方法中定义的本地变量或方法的参数,则变量必须被声明 final。2) 内部类可以声明为 private 或 protected;还可以声明为 abstract 或 final。3) 内部类可以声明为 static 的,但此时就不能再使用外部类的非 static 的成员变量和非static 的成员方法;4) 非 static 的内部类中的成员不能声明为 static 的,只有在顶层类或 static 的内部类中才可声明 static 成员。我们为什么使用内部类1、在
20、内部类(inner class)中,可以随意的访问外部类的成员,这可以让我们更好地组织管理我们的代码,增强代码的可读性。2、内部类可以用于创建适配器类,适配器类是用于实现接口的类。使用内部类来实现接口,可以更好地定位与接口关联的方法在代码中的位置。3、内部类的更多用法:见下面代码下面为通过内部类来实现接口的简单代码:interface Animalvoid eat();void sleep();class Zooprivate class Tiger implements Animal/通过内部类 Tiger 来实现 Animal 接口,并且声明为 private.从而可以隐藏接口内部的具体实
21、现public void eat()System.out.println(“tiger eat“);public void sleep()System.out.println(“tiger sleep“);Tiger gettiger()return new Tiger();class Testpublic static void main(String args)Zoo zoo =new Zoo();Animal an=zoo.gettiger();an.eat();an.sleep();下面通过内部类来解决同时实现继承和接口,但是要继承的类中的方法名和接口中的方法名是一样的,此时可以通过内
22、部类来实现接口中的方法,达到区分同名方法的目的:interface Machine void run();class Personvoid run()System.out.println(“people are running“);class Robet extends Person/实现继承private class MachineStart implements Machine/public void run()System.out.println(“machine started“);MachineStart getmachine()return new MachineStart();class Testpublic static void main(String args)Robet robet=new Robet();Machine machine=robet.getmachine();machine.run();robet.run();