Java内部类详解

转自https://www.cnblogs.com/dolphin0520/p/3811445.html  谢谢

 一.内部类基础

  二.深入理解内部类

  三.内部类的使用场景和好处

  四.常见的与内部类相关的笔试面试题

  若有不正之处,请多谅解并欢迎批评指正。

  请尊重作者劳动成果,转载请标明原文链接:

  http://www.cnblogs.com/dolphin0520/p/3811445.html

一.内部类基础

  在Java中,可以将一个类定义在另一个类里面或者一个方法里面,这样的类称为内部类。广泛意义上的内部类一般来说包括这四种:成员内部类、局部内部类、匿名内部类和静态内部类。下面就先来了解一下这四种内部类的用法。

  1.成员内部类

  成员内部类是最普通的内部类,它的定义为位于另一个类的内部,形如下面的形式:

1
2
3
4
5
6
7
8
9
10
11
12
13
class  Circle {
     double  radius =  0 ;
     
     public  Circle( double  radius) {
         this .radius = radius;
     }
     
     class  Draw {      //内部类
         public  void  drawSahpe() {
             System.out.println( "drawshape" );
         }
     }
}

  这样看起来,类Draw像是类Circle的一个成员,Circle称为外部类。成员内部类可以无条件访问外部类的所有成员属性和成员方法(包括private成员和静态成员)。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
class  Circle {
     private  double  radius =  0 ;
     public  static  int  count = 1 ;
     public  Circle( double  radius) {
         this .radius = radius;
     }
     
     class  Draw {      //内部类
         public  void  drawSahpe() {
             System.out.println(radius);   //外部类的private成员
             System.out.println(count);    //外部类的静态成员
         }
     }
}

  不过要注意的是,当成员内部类拥有和外部类同名的成员变量或者方法时,会发生隐藏现象,即默认情况下访问的是成员内部类的成员。如果要访问外部类的同名成员,需要以下面的形式进行访问:

1
2
外部类. this .成员变量
外部类. this .成员方法

  虽然成员内部类可以无条件地访问外部类的成员,而外部类想访问成员内部类的成员却不是这么随心所欲了。在外部类中如果要访问成员内部类的成员,必须先创建一个成员内部类的对象,再通过指向这个对象的引用来访问:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class  Circle {
     private  double  radius =  0 ;
 
     public  Circle( double  radius) {
         this .radius = radius;
         getDrawInstance().drawSahpe();    //必须先创建成员内部类的对象,再进行访问
     }
     
     private  Draw getDrawInstance() {
         return  new  Draw();
     }
     
     class  Draw {      //内部类
         public  void  drawSahpe() {
             System.out.println(radius);   //外部类的private成员
         }
     }
}

  成员内部类是依附外部类而存在的,也就是说,如果要创建成员内部类的对象,前提是必须存在一个外部类的对象。创建成员内部类对象的一般方式如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
public  class  Test {
     public  static  void  main(String[] args)  {
         //第一种方式:
         Outter outter =  new  Outter();
         Outter.Inner inner = outter. new  Inner();   //必须通过Outter对象来创建
         
         //第二种方式:
         Outter.Inner inner1 = outter.getInnerInstance();
     }
}
 
class  Outter {
     private  Inner inner =  null ;
     public  Outter() {
         
     }
     
     public  Inner getInnerInstance() {
         if (inner ==  null )
             inner =  new  Inner();
         return  inner;
     }
      
     class  Inner {
         public  Inner() {
             
         }
     }
}
      内部类可以拥有private访问权限、protected访问权限、public访问权限及包访问权限。比如上面的例子,如果成员内部类Inner用private修饰,则只能在外部类的内部访问,如果用public修饰,则任何地方都能访问;如果用protected修饰,则只能在同一个包下或者继承外部类的情况下访问;如果是默认访问权限,则只能在同一个包下访问。这一点和外部类有一点不一样,外部类只能被public和包访问两种权限修饰。我个人是这么理解的,由于成员内部类看起来像是外部类的一个成员,所以可以像类的成员一样拥有多种权限修饰。

     

2.局部内部类

  局部内部类是定义在一个方法或者一个作用域里面的类,它和成员内部类的区别在于局部内部类的访问仅限于方法内或者该作用域内。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class  People{
     public  People() {
         
     }
}
 
class  Man{
     public  Man(){
         
     }
     
     public  People getWoman(){
         class  Woman  extends  People{    //局部内部类
             int  age = 0 ;
         }
         return  new  Woman();
     }
}

  注意,局部内部类就像是方法里面的一个局部变量一样,是不能有public、protected、private以及static修饰符的。

 3.静态内部类

  静态内部类也是定义在另一个类里面的类,只不过在类的前面多了一个关键字static。静态内部类是不需要依赖于外部类的,这点和类的静态成员属性有点类似,并且它不能使用外部类的非static成员变量或者方法,这点很好理解,因为在没有外部类的对象的情况下,可以创建静态内部类的对象,如果允许访问外部类的非static成员就会产生矛盾,因为外部类的非static成员必须依附于具体的对象。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public  class  Test {
     public  static  void  main(String[] args)  {
         Outter.Inner inner =  new  Outter.Inner();
     }
}
 
class  Outter {
     public  Outter() {
         
     }
     
     static  class  Inner {
         public  Inner() {
             
         }
     }
}

 

package net.xsoftlab.baike;

public class Circle {
    double radius=0;
    public static int count =1;
    public void circle(double radius)
    {
       this.radius=radius;
    }
   static class Inner{
        public  void  drawshape()
        {
            System.out.println(radius);//报错,由于调用了非静态成员
            System.out.println(count);
        }
    }
}

二.深入理解内部类

 1.为什么成员内部类可以无条件访问外部类的成员?

  在此之前,我们已经讨论过了成员内部类可以无条件访问外部类的成员,那具体究竟是如何实现的呢?下面通过反编译字节码文件看看究竟。事实上,编译器在进行编译的时候,会将成员内部类单独编译成一个字节码文件,下面是Outter.java的代码:


三.内部类的使用场景和好处

为什么在Java中需要内部类?总结一下主要有以下四点:

  1.每个内部类都能独立的继承一个接口的实现,所以无论外部类是否已经继承了某个(接口的)实现,对于内部类都没有影响。内部类使得多继承的解决方案变得完整,

  2.方便将存在一定逻辑关系的类组织在一起,又可以对外界隐藏。

  3.方便编写事件驱动程序

  4.方便编写线程代码

  个人觉得第一点是最重要的原因之一,内部类的存在使得Java的多继承机制变得更加完善。

四.常见的与内部类相关的笔试面试题

 1.根据注释填写(1),(2),(3)处的代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public  class  Test{
     public  static  void  main(String[] args){
            // 初始化Bean1
            ( 1 )
            bean1.I++;
            // 初始化Bean2
            ( 2 )
            bean2.J++;
            //初始化Bean3
            ( 3 )
            bean3.k++;
     }
     class  Bean1{
            public  int  I =  0 ;
     }
 
     static  class  Bean2{
            public  int  J =  0 ;
     }
}
 
class  Bean{
     class  Bean3{
            public  int  k =  0 ;
     }
}

  从前面可知,对于成员内部类,必须先产生外部类的实例化对象,才能产生内部类的实例化对象。而静态内部类不用产生外部类的实例化对象即可产生内部类的实例化对象。

  创建静态内部类对象的一般形式为:  外部类类名.内部类类名 xxx = new 外部类类名.内部类类名()

  创建成员内部类对象的一般形式为:  外部类类名.内部类类名 xxx = 外部类对象名.new 内部类类名()

  

 因此,(1),(2),(3)处的代码分别为:

Test test = new Test();    

  Test.Bean1 bean1 = test.new Bean1();   

 

Test.Bean2 b2 = new Test.Bean2();    

 

Bean bean = new Bean();     

Bean.Bean3 bean3 =  bean.new Bean3();   

2.下面这段代码的输出结果是什么?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public  class  Test {
     public  static  void  main(String[] args)  {
         Outter outter =  new  Outter();
         outter. new  Inner().print();
     }
}
 
 
class  Outter
{
     private  int  a =  1 ;
     class  Inner {
         private  int  a =  2 ;
         public  void  print() {
             int  a =  3 ;
             System.out.println( "局部变量:"  + a);
             System.out.println( "内部类变量:"  this .a);
             System.out.println( "外部类变量:"  + Outter. this .a);
         }
     }
}
3
2
1

 

最后补充一点知识:关于成员内部类的继承问题。一般来说,内部类是很少用来作为继承用的。但是当用来继承的话,要注意两点:

  1)成员内部类的引用方式必须为 Outter.Inner.

  2)构造器中必须有指向外部类对象的引用,并通过这个引用调用super()。这段代码摘自《Java编程思想》

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class  WithInner {
     class  Inner{
         
     }
}
class  InheritInner  extends  WithInner.Inner {
      
     // InheritInner() 是不能通过编译的,一定要加上形参
     InheritInner(WithInner wi) {
         wi. super ();  //必须有这句调用
     }
  
     public  static  void  main(String[] args) {
         WithInner wi =  new  WithInner();
         InheritInner obj =  new  InheritInner(wi);
     }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值