前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java-内部类详解

Java-内部类详解

作者头像
Fisherman渔夫
发布2019-07-31 15:47:05
3220
发布2019-07-31 15:47:05
举报
文章被收录于专栏:渔夫渔夫

版权声明:署名,允许他人基于本文进行创作,且必须基于与原先许可协议相同的许可协议分发本文 (Creative Commons

Java内部类引言

 说起内部类这个词,想必很多人都不陌生,但是又会觉得不熟悉。原因是平时编写代码时可能用到的场景不多,用得最多的是在有事件监听的情况下,并且即使用到也很少去总结内部类的用法。今天我们就来一探究竟。下面是本文的目录大纲:

 一.内部类基础

 二.深入理解内部类

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

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

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

一、内部类基础

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

1.成员内部类

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

Demo-1:

代码语言:javascript
复制
class Circle {
    double radius = 0;
     
    public Circle(double radius) {
        this.radius = radius;
    }
     
    class Draw {     //内部类
        public void drawSahpe() {
            System.out.println(radius);
    }
}

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

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

代码语言:javascript
复制
外部类.this.成员变量
外部类.this.成员方法

 而在内部类中使用 this.成员变量/方法,代表的是调用内部类自身的成员变量/方法。

 虽然成员内部类可以无条件地访问外部类的成员,而外部类想访问成员内部类的成员却不是这么随心所欲了。在外部类中如果要访问成员内部类的成员,必须先创建一个成员内部类的对象(在此描述的是非静态内部类的性质,静态内部类下面章节会对静态内部类进行专门性质讲解),再通过指向这个对象的引用来访问:

Demo-2:

代码语言:javascript
复制
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成员
        }
    }
}

成员内部类是依附外部类而存在的(如果在父类的构造器中创建了内部类对象,那么其可能造成父类this对象的逸出(链接:xxx)),也就是说,如果要创建成员内部类的对象,前提是必须存在一个外部类的对象。反过来说,创建外部类后内部类并不是自动就创建完成了。创建成员内部类对象的一般方式如下:

Demo-3:

代码语言:javascript
复制
public class Test {
    public static void main(String[] args)  {
        //第一种方式:先创建外部类对象,再调用外部类对象中的内部类构造方法构造内部类对象
        Outter outter = new Outter();
        Outter.Inner inner = outter.new Inner();  //必须通过Outter对象来创建
         
        //第二种方式:还是先创建外部类对象,再调用外部类返回内部类的getInnerInstance(),其实质上也是调用了内部类的构造器
        Outter outter2 = new Outter();
        Outter.Inner inner1 = outter2.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修饰,则只能在同一个包下或者继承外部类的情况下访问, 即:

代码语言:javascript
复制
/**
*将前面一个代码内部类默认defulat修饰改为private修饰,那么就不能在其他类中main方法中构造内部类对象
/
 Outter outter = new Outter();
 Outter.Inner inner = outter.new Inner();  //报错,不能访问私有内部类

 如果是默认访问权限,则只能在同一个包下访问。这一点和外部类有一点不一样,外部类只能被public和包访问两种权限修饰。我个人是这么理解的,由于成员内部类看起来像是外部类的一个成员,所以可以像类的成员一样拥有多种权限修饰,不过有一个显著的区别,外部类构造器默认不会构造内部类。

2.局部内部类

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

Demo-4:

代码语言:javascript
复制
class People{
    public People() {
         
    }
}
 
class Man{
    public Man(){
         
    }
     
    public People getWoman(){
        class Woman extends People{   //局部内部类Woman定义在Man类的方法中
            int age =0;
        }
        return new Woman();
    }
}

 注意,局部内部类与局部变量有一样的性质,即:也不能被public、protected、private以及static修饰

3.匿名内部类

 匿名内部类应该是平时我们编写代码时用得最多的,在编写事件监听的代码时使用匿名内部类不但方便,而且使代码更加容易维护。Demo-5是一段Android事件监听代码:

Demo-5:

代码语言:javascript
复制
can_bt.setOnClickListener(new OnClickListener() {
             
            @Override
            public void onClick(View v) {
                // TODO Auto-generated method stub
                 
            }
        });
         
        history_bt.setOnClickListener(new OnClickListener() {
             
            @Override
            public void onClick(View v) {
                // TODO Auto-generated method stub
                 
            }
        });

 这段代码为两个按钮设置监听器,这里面就使用了匿名内部类。这段代码中的:

代码语言:javascript
复制
new OnClickListener() {
             
            @Override
            public void onClick(View v) {
                // TODO Auto-generated method stub
                 
            }
        }

 就是匿名内部类的使用。代码中需要给按钮设置监听器对象,使用匿名内部类能够在实现父类或者接口中的方法情况下同时产生一个相应的对象,但是前提是这个父类或者接口必须先存在才能这样使用。

 当然像Demo-6中的写法也是可以的,跟上面使用匿名内部类达到效果相同。

Demo-6:

代码语言:javascript
复制
private void setListener()
{
    scan_bt.setOnClickListener(new Listener1());       
    history_bt.setOnClickListener(new Listener2());
}
 
class Listener1 implements View.OnClickListener{
    @Override
    public void onClick(View v) {
    // TODO Auto-generated method stub
             
    }
}
 
class Listener2 implements View.OnClickListener{
    @Override
    public void onClick(View v) {
    // TODO Auto-generated method stub
             
    }
}

 这种写法虽然能达到一样的效果,但是既冗长又难以维护,所以一般使用匿名内部类的方法来编写事件监听代码。同样的,匿名内部类也是不能有访问修饰符和static修饰符的。

 匿名内部类是唯一一种没有构造器的类,因为其可以认为是对父类/接口构造器的单独一次重写(构造方法在继承中是不能被重写的)。正因为其没有构造器,所以匿名内部类的使用范围非常有限,大部分匿名内部类用于接口回调。匿名内部类在编译的时候由系统自动起名为Outter$1.class。一般来说,匿名内部类用于继承其他类或是实现接口,并不需要增加额外的方法,只是对继承方法的实现或是重写。

 关于匿名内部类的友情链接:https://blog.csdn.net/li_xunhuan/article/details/96435905

4.静态内部类

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

Demo-7:

代码语言:javascript
复制
public class Test {
    public static void main(String[] args)  {
   //注意一旦用静态修饰内部类,那么不需要构建外部类对象,再构建内部类对象;
        Outter.Inner inner = new Outter.Inner();
    }
}
 
class Outter {    
    public Outter() {
         
    }
     
    static class Inner {
        public Inner() {
             
        }
    }
}

二.深入理解内部类

 参考网址:https://www.cnblogs.com/dolphin0520/p/3811445.html

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

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

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

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

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

 4.方便编写线程代码

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

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

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

 2)构造器中必须有指向外部类对象的引用,其目的是通过这个引用调用super()。

   Demo-8是补充知识的例子,摘自《Java编程思想》:

Demo-8:

代码语言:javascript
复制
class WithInner {
    class Inner{
         
    }
}
class InheritInner extends WithInner.Inner {//继承中:成员内部类的引用方式必须为 Outter.Inner.
      
    // InheritInner() 是不能通过编译的,一定要加上外部类对象的形参,为了调用外部类的构造方法
    InheritInner(WithInner wi) {
        wi.super(); //必须有这句调用
    }
  
    public static void main(String[] args) {
        WithInner wi = new WithInner();
        InheritInner obj = new InheritInner(wi);
    }
} 

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2019年07月27日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • Java内部类引言
  • 一、内部类基础
    • 1.成员内部类
      • 2.局部内部类
        • 3.匿名内部类
          • 4.静态内部类
          • 二.深入理解内部类
          • 三.内部类的使用场景和好处
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档