一。内部类基础
二。深入理解内部类
三。内部类的使用场景和好处
四。常见的与内部类相关的笔试面试题
若有不正之处,请多谅解并欢迎批评指正。
一。内部类基础
在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。广泛意义上的内部类一般来说包括这四种:成员内部类、局部内部类、匿名内部类和静态内部类。下面就先来了解一下这四种内部类的用法。
1.成员内部类
成员内部类是最普通的内部类,它的定义为位于另一个类的内部,形如下面的形式:
class Circle { doubleradius = 0; publicCircle( doubleradius) { this.radius = radius; } class Draw { //内部类publicvoiddrawSahpe() { System. out.println( “drawshape”); } } }
这样看起来,类Draw像是类Circle的一个成员,Circle称为外部类。成员内部类可以无条件访问外部类的所有成员属性和成员方法(包括private成员和静态成员)。
class Circle { privatedoubleradius = 0; publicstaTIcintcount = 1; publicCircle( doubleradius) { this.radius = radius; } class Draw { //内部类publicvoiddrawSahpe() { System.out.println(radius); //外部类的private成员System. out.println(count); //外部类的静态成员} } }
不过要注意的是,当成员内部类拥有和外部类同名的成员变量或者方法时,会发生隐藏现象,即默认情况下访问的是成员内部类的成员。如果要访问外部类的同名成员,需要以下面的形式进行访问:
外部类。 this.成员变量 外部类。 this.成员方法
虽然成员内部类可以无条件地访问外部类的成员,而外部类想访问成员内部类的成员却不是这么随心所欲了。在外部类中如果要访问成员内部类的成员,必须先创建一个成员内部类的对象,再通过指向这个对象的引用来访问:
class Circle { privatedoubleradius = 0; publicCircle( doubleradius) { this.radius = radius; getDrawInstance().drawSahpe(); //必须先创建成员内部类的对象,再进行访问}privateDraw getDrawInstance() { returnnewDraw(); } class Draw { //内部类publicvoiddrawSahpe() { System. out.println(radius); //外部类的private成员} } }
成员内部类是依附外部类而存在的,也就是说,如果要创建成员内部类的对象,前提是必须存在一个外部类的对象。创建成员内部类对象的一般方式如下:
publicclassTest { publicstaTIcvoidmain(String[] args) { //第一种方式:Outter outter =newOutter(); Outter.Inner inner = outter. newInner(); //必须通过Outter对象来创建//第二种方式:Outter.Inner inner1 = outter.geTInnerInstance(); } } class Outter { privateInner inner = null; publicOutter() { } publicInner geTInnerInstance() { if(inner == null) inner =newInner(); returninner; } class Inner { publicInner() { } } }
内部类可以拥有private访问权限、protected访问权限、public访问权限及包访问权限。比如上面的例子,如果成员内部类Inner用private修饰,则只能在外部类的内部访问,如果用public修饰,则任何地方都能访问;如果用protected修饰,则只能在同一个包下或者继承外部类的情况下访问;如果是默认访问权限,则只能在同一个包下访问。这一点和外部类有一点不一样,外部类只能被public和包访问两种权限修饰。我个人是这么理解的,由于成员内部类看起来像是外部类的一个成员,所以可以像类的成员一样拥有多种权限修饰。
2.局部内部类
局部内部类是定义在一个方法或者一个作用域里面的类,它和成员内部类的区别在于局部内部类的访问仅限于方法内或者该作用域内。
classPeople{publicPeople() { } } classMan{publicMan(){ } publicPeople getWoman(){classWomanextendsPeople{//局部内部类intage = 0; } returnnewWoman(); } }
注意,局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及static修饰符的。
3.匿名内部类
匿名内部类应该是平时我们编写代码时用得最多的,在编写事件监听的代码时使用匿名内部类不但方便,而且使代码更加容易维护。下面这段代码是一段Android事件监听代码:
scan_bt.setOnClickListener( newOnClickListener() { @OverridepublicvoidonClick(View v) { // TODO Auto-generated method stub} }); history_bt.setOnClickListener(newOnClickListener() { @OverridepublicvoidonClick(View v) { // TODO Auto-generated method stub} });
这段代码为两个按钮设置监听器,这里面就使用了匿名内部类。这段代码中的:
newOnClickListener() { @OverridepublicvoidonClick(View v) { // TODO Auto-generated method stub} }
就是匿名内部类的使用。代码中需要给按钮设置监听器对象,使用匿名内部类能够在实现父类或者接口中的方法情况下同时产生一个相应的对象,但是前提是这个父类或者接口必须先存在才能这样使用。当然像下面这种写法也是可以的,跟上面使用匿名内部类达到效果相同。
privatevoidsetListener() { scan_bt.setOnClickListener( newListener1()); history_bt.setOnClickListener( newListener2()); }classListener1implementsView.OnClickListener{@Override publicvoidonClick(View v) { // TODO Auto-generated method stub} }classListener2implementsView.OnClickListener{@Override publicvoidonClick(View v) { // TODO Auto-generated method stub} }
这种写法虽然能达到一样的效果,但是既冗长又难以维护,所以一般使用匿名内部类的方法来编写事件监听代码。同样的,匿名内部类也是不能有访问修饰符和static修饰符的。
匿名内部类是唯一一种没有构造器的类。正因为其没有构造器,所以匿名内部类的使用范围非常有限,大部分匿名内部类用于接口回调。匿名内部类在编译的时候由系统自动起名为Outter$1.class。一般来说,匿名内部类用于继承其他类或是实现接口,并不需要增加额外的方法,只是对继承方法的实现或是重写。
4.静态内部类
静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字static。静态内部类是不需要依赖于外部类的,这点和类的静态成员属性有点类似,并且它不能使用外部类的非static成员变量或者方法,这点很好理解,因为在没有外部类的对象的情况下,可以创建静态内部类的对象,如果允许访问外部类的非static成员就会产生矛盾,因为外部类的非static成员必须依附于具体的对象。
publicclassTest { publicstaticvoidmain(String[] args) { Outter.Inner inner =newOutter.Inner(); } } class Outter { publicOutter() { } staticclass Inner { publicInner() { } } }