【JavaSE】玩转内部类--内部类的分类及其不为人知的细节

一、内部类的概念

内部类的是定义在另一个类中的类

一般分为:

  1. 本地内部类
  2. 实例内部类
  3. 静态内部类
  4. 匿名内部类

二、内部类的分类

1.本地内部类

定义在方法当中的类。

public class TestDemo {
    public void func(){
        class  Test{
            public int a;
        }
        //缺点:只能在当前方法中使用——本地内部类,定义在方法内部的类
    }

本地内部类,我们用得比较少,我们写本地内部类的时候一定要谨慎。

同时,本地内部类有一个很大的缺点:只能在当前方法中使用。

2.实例内部类

实例内部类是指没有用 static 修饰的内部类,有的地方也称为非静态内部类。

简单示例如下:

public class OuterClass {
    class InnerClass {
        // 实例内部类
    }
}

上面的InnerClass就是实力内部类了。

实例内部类的注意事项有以下几点,我们通过实例来一一了解:

  1. 在实例内部类中不能定义一个静态的成员变量,如果非要定义,只能定义一个静态常量,也就是用final修饰。
class OuterClass{//定义了一个普通类
    //实例内部类,可以把他当作外部类的一个普通实例的成员
    class InnerClass{
        public int data4=4;
        private int data5 = 5;
        //public static int data6 = 6; //err,不能定义静态成员变量,因为如果一个东西是静态的,它是属于类的,不属于对象的
        //加final就可以,因为加了final,它就变成了一个静态常量
        public static final int data6 = 6;
        }
     }
  1. 我们如何实例化 实例内部类的对象?

外部类名.内部类名 变量 = 外部类对象的引用.new 内部类();

class OuterClass{//定义了一个普通类
    //实例内部类,可以把他当作外部类的一个普通实例的成员
    class InnerClass{
        public int data4=4;
        private int data5 = 5;
        //public static int data6 = 6; //err,不能定义静态成员变量,因为如果一个东西是静态的,它是属于类的,不属于对象的
        //加final就可以,因为加了final,它就变成了一个静态常量
        public static final int data6 = 6;
        }
     }

public static void main(String[] args) {
        OuterClass outerClass = new OuterClass();
        //OuterClass.InnerClass innerClass = new OuterClass.InnerClass();//ERR ,直接new是不能编译通过
        //需要外部类对象的引用才能实例化一个内部类对象
        OuterClass.InnerClass innerClass =  outerClass.new InnerClass();
    }
  1. 实例内部类当中,如果包含了和外部类同名的成员变量,我们如何在实例内部类中访问?

我们可以通过外部类的this来访问。注意:我们实例内部类当中包含了两个this,一个是外部类的this,一个是内部类的this。

//实例内部类
class OuterClass{//定义了一个普通类
    public int data1= 1;

    //实例内部类,可以把他当作外部类的一个普通实例的成员
    class InnerClass{
        public int data1=999;
        public InnerClass(){
            System.out.println("不带参数的内部类构造方法");
        }
        public void test(){
            System.out.println(OuterClass.this.data1);//1
            System.out.println(this.data1);//999
        }

    }
}


public class TestDemo {
    public static void main(String[] args) {
    //创建内部类对象分两步,1:
        OuterClass outerClass = new OuterClass();
//2.实例化一个内部类对象
        OuterClass.InnerClass innerClass =  outerClass.new InnerClass();
        innerClass.test();
    }
}

在这里插入图片描述

3.静态内部类

实例内部类是指用 static 修饰的内部类

public class OuterClass {
    static class InnerClass {
        // 静态内部类
    }
}

静态内部类的一些问题:

  1. 如何访问外部类的普通成员变量?

静态内部类不能访问外部类的实例成员,只能访问外部类的类成员。

 class OuterClass2{
    public int data1 = 1;
    private int data2 = 2;
    public static int data3 = 3;
    //静态内部类
    static class InnerClass{
        public int data4 = 4;
        private int data5 = 5;
        public static int data6 = 6;

        //写一个构造方法
        public OuterClass2 out;
        public InnerClass(OuterClass2 out){
            this.out = out;
        }

        public void test(){
            //System.out.println(data1); //err
            System.out.println(out.data1);
            //System.out.println(data2);//err
            //System.out.println(new OuterClass2().data2); //由于是私有的,访问不了,要访问,只能提供get()和set()方法了
            System.out.println(data3);
            System.out.println(data4);
            System.out.println(data5);
            System.out.println(data6);
            System.out.println("InnerClass::test()");
        }
    }
  1. 如何实例化静态成员变量?

外部类名.内部类名 变量 =new 外部类对象的引用.内部类(变量);

public static void main(String[] args) {
//这里有两步 
//1.实例化外部类类对象
        OuterClass2 outerClass2 = new OuterClass2();
//2.通过外部类对象的引用实例化一个内部类对象
        OuterClass2.InnerClass innerClass = new OuterClass2.InnerClass(outerClass2);
        innerClass.test();
    }

打印结果:

在这里插入图片描述

3.匿名内部类

名内部类就是一个没有显式的名字的内部类。
匿名内部类没有类名,因此就不可能有构造函数,不能创建对象。

语法一:

new 实现接口()
{
    //匿名内部类类体部分
}

语法二:

new 父类构造器(实参列表)
{
  //匿名内部类类体部分
}

实例:

class Test{
    public void test(){
        System.out.println("test()haha!");
    }
}

public class TestDemo2 {
    public static void main(String[] args) {
        new Test(){
            @Override
            public void test(){
                System.out.println("我是重写的test方法");
            }
        }.test();//调用test()方法
    }

就是定义了一个类Test,在类里定义了一个test方法。调用他的实例方法test()。

然后调用它的test()方法。

在调用test()方法的同时,重写了test()方法,所以打印结果为:我是重写的test方法。

输出结果:
在这里插入图片描述
所以,匿名内部类的最大好处就是可以简化我们代码的书写。


总结

为什么需要使用内部类呢? 其主要原因有以下三点:
•内部类方法可以访问该类定义所在的作用域中的数据, 包括私有的数据。

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

•当想要定义一个回调函数且不想编写大量代码时,使用匿名 (anonymous) 内部类比较便捷。

  • 28
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 26
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 26
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

十叶知秋

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值