【JavaSE】内部类

目录

前言

内部类

内部类的种类

1. 实例内部类

2 静态内部类

3 匿名内部类

4 局部内部类

结语


前言

内部类是我们前面学习遗留下来的知识点,在学完接口后才能更好的理解它,因此等到现在才讲

内部类

在Java中,我们可以将A类定义在B类或者一个方法的内部,这时A类就叫做内部类,B类就叫做外部类。内部类其实也是封装的一种体现。

class B {
    //...
    class A {
        //...
    }
}

内部类和外部类是共用一个Java源文件的,但是在编译后,内部类会形成单独的字节码文件(文件名为:外部类类名$内部类类名.class)


内部类的种类

  1. 实例内部类:非static修饰的成员内部类
  2. 静态内部类:static修饰的成员内部类
  3. 匿名内部类:与接口相关的内部类
  4. 局部内部类:定义在外部类的方法或{ }中

相比较于其他三种内部类,局部内部类用的很少,下面我们来一一认识这些内部类


1. 实例内部类

定义在类里面,方法外面,且不被static修饰。在实例内部类中,我们也可以正常定义成员变量和成员方法,静态的也可以

class OuterClass {
    public int O1;
    private int O2;
    public static int O3;

    /**
     * 实例内部类:定义在类里面,方法外面
     */
    class InnerClass {
        public int I1;
        private int I2;
        public static int I3;

        public void testI() {
            System.out.println("InnerClass::testI()");
        }
    }

    public void testO() {
        System.out.println("OuterClass::testO()");
    }
}

接着我们来实例化内部类对象,它的语法格式如下:

方法一:外部类类名.内部类类名 变量名 = new 外部类类名( ).new 内部类类名( )

方法二:先 外部类类名.变量名1 = new 外部类类名( );

              再 外部类类名.内部类类名 变量名2 = 变量名1.new 内部类类名( )

public class Test {
    public static void main(String[] args) {
        //方法一:
        OuterClass.InnerClass innerClass = new OuterClass().new InnerClass();
        //方法二:
        OuterClass outerClass = new OuterClass();
        OuterClass.InnerClass innerClass1 = outerClass.new InnerClass();
    }
}

总而言之,我们必须得通过外部类对象的引用再. new一个内部类对象


  • 创建好内部类对象后,我们就可以使用它来调用各种成员
        System.out.println(innerClass.I1);//调用内部类的成员
        innerClass.testI();//调用内部类的方法
  • 其实我们也可以在内部类方法中直接调用外部类的任何成员
        public void testI() {
            System.out.println("InnerClass::testI()");
            System.out.println(I1);
            System.out.println(O1);
        }
  •  当内部类和外部类存在同名成员变量时,内部类方法中调用时会优先访问内部类的成员,
     那如果我们想要精准的访问(同名)内部类成员和(同名)外部类成员要怎么做呢
     答案就是:this.同名变量 和 外部类名.this.同名变量
class OuterClass {
    public int same = 100;

    class InnerClass {
        public int same = 10;

        public void testI() {
            System.out.println("内部类"+this.same);
            System.out.println("外部类"+OuterClass.this.same);
        }
    }
}
public class Test {
    public static void main(String[] args) {
        OuterClass.InnerClass innerClass = new OuterClass().new InnerClass();
        innerClass.testI();
    }
}

运行结果如下

或者我们也可以在内部类中创建一个外部类的对象,再用外部类的对象去调用外部类的成员

  • 外部类中,不能直接访问内部类中的成员,如果像访问就必须得先创建内部类的成员

2 静态内部类

定义在类里面,方法外面,且被static修饰。在静态内部类中,我们也可以正常定义成员变量和成员方法,静态的也可以

class OuterClass {
    public int O1;
    private int O2;
    public static int O3;

    /**
     * 静态内部类:定义在类里面,方法外面,且有static修饰
     */
    static class InnerClass {
        public int I1;
        private int I2;
        public static int I3;

        public void testI() {
            System.out.println("InnerClass::testI()");
        }
    }

    public void testO() {
        System.out.println("OuterClass::testO()");
    }
}

 接着我们来实例化内部类对象,它的语法格式如下:

外部类类名.内部类类名 变量名 = new 外部类类名.内部类类名( )

(创建静态内部类对象时,不需要先创建外部类对象)

public class Test {
    public static void main(String[] args) {
        OuterClass.InnerClass innerClass = new OuterClass.InnerClass();
    }
}

其他的东西跟实例内部类非常相似,但是在静态内部类中调用外部类非静态成员变量时,我们需要先创建外部类对象,用该对象才能引用

class OuterClass {
    public int O1;

    static class InnerClass {
        public int I1;

        public void testI() {
            OuterClass outerClass = new OuterClass();
            System.out.println(outerClass.O1);
        }
    }
}
public class Test {
    public static void main(String[] args) {
        OuterClass.InnerClass innerClass = new OuterClass.InnerClass();
    }
}

3 匿名内部类

在将匿名内部类前,我们先来看一下什么叫做匿名对象

//方法1:
        OuterClass outerClass = new OuterClass();
        outerClass.testO();
        System.out.println(outerClass.O1);
//方法2:
        new OuterClass().testO();
        System.out.println(new OuterClass().O1);

在方法1中,我们使用OuterClass引用创建了一个outerClass对象,打印O1和调用testO( )方法

在方法2中,我们使用new OuterClass( )直接打印O1和调用testO( )方法

方法2中每次都会创建一个新的对象,它没有用变量去承接对象的地址,此时这个对象就叫做匿名对象


匿名内部类跟接口有关,我们都知道,接口不能直接创建对象。我们通常会用一个类去实现接口,重写方法,这样才能实例化类的对象,从而使用接口。而匿名内部类就这个过程有点相似

interface ITest {
    void test();
}

public class Test {
    public static void main(String[] args) {
        new ITest() {
            @Override
            public void test() {
                System.out.println("重写后的test()");
            }
        };
    }
}

其中,这部分就叫做匿名内部类

它并没有真正实例化接口对象;它等同于一个类实现了这个接口,同时重写了test方法,但是这个类我们并不知道它叫什么,这就是匿名内部类


那如果我们想要调用test方法,要怎么做呢?

方法一:可以直接在匿名内部类后面. test( )

        new ITest() {
            @Override
            public void test() {
                System.out.println("重写后的test()");
            }
        }.test();

 

方法二:创建一个匿名内部类对象,再用对象去调用test方法

        ITest iTest = new ITest() {
            @Override
            public void test() {
                System.out.println("重写后的test()");
            }
        };
        iTest.test();

 


4 局部内部类

定义在方法里的类,即为局部的局部,只能在当前方法里使用,出了方法后就没用了,局限性很大

class OuterClass {
    public int O1;

    public void testO() {
        class InnerClass {
            public int I1;

            public void testI() {
                System.out.println("InnerClass::testI()");
            }
        }
    }
}

它不能被public、static修饰;它也有自己独立的字节码文件,格式为:外部类类名$数字+内部类类名.class

因为局限性太大,我们几乎不会使用它,了解即可 

结语

关于内部类我们就讲到这里。希望大家能喜欢这篇文章,有总结不到位的地方还请多多谅解,若有出现纰漏,希望大佬们看到错误之后能够在私信或评论区指正,博主会及时改正,共同进步!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值