内部类详解

简介: 内部类详解

一、什么是内部类

定义:当一个事物的内部,还有一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么这个内部的完整结构做好使用内部类。在Java中,可以将一个类定义在另一个类或者一个

public class OutClass {
class InnerClass{
}
}
// OutClass是外部类
// InnerClass是内部类

方法的内部,前者成为内部类,后者成为外部类。内部类也是封装的一种体现

【注意事项】

1.定义在class类名{}花括号外部的,即使是在一个文件里,都不能成为内部类

public class A{
}
 
class B{
}
 
//A 和 B是两个独立的类,彼此之间没有关系。

2. 内部类和外部类共用同一个java源文件,但是经过编译之后,内部类会形成单独的字节码文件

二、内部类的划分

在外部类中,内部类定义位置与外部类成员所处的位置相同,因此称为成员内部类

2.1 实例内部类

即为被static修饰的成员内部类。

class OutClass {
    private int a;
    static int b;
    int c;
 
    public void method() {
        a = 10;
        System.out.println(a);
    }
 
    public static void methodB() {
        System.out.println(b);
    }
 
    //    实例内部类:未被static修饰
    class InnerClass {
        //在实例内部类中可以直接访问外部类中:任意访问限定符修饰的成员
        int c;
 
        public void methodInner() {
            a = 100;
            b = 200;
            method();
            methodB();
//如果外部类和实例内部类中具有相同名称成员时,优先访问的是内部类自己的
            c = 300;
            System.out.println(c);
            //如果要访问外部类同名成员的时候,必须外部类名称.this.同名成员名字
            OutClass.this.c = 400;
            System.out.println(OutClass.this.c);
            System.out.println(c);
        }
    }
    public static void main(String[] args) {
// 外部类:对象创建 以及 成员访问
        OutClass outClass = new OutClass();
        System.out.println(outClass.a);
        System.out.println(OutClass.b);
        System.out.println(outClass.c);
        outClass.method();
        outClass.methodB();
        System.out.println("=============实例内部类的访问=============");
// 要访问实例内部类中成员,必须要创建实例内部类的对象
// 而普通内部类定义与外部类成员定义位置相同,因此创建实例内部类对象时必须借助外部类
// 创建实例内部类对象
        OutClass.InnerClass innerClass1 = new OutClass().new InnerClass();
// 上述语法比较怪异,也可以先将外部类对象先创建出来,然后再创建实例内部类对象
        OutClass.InnerClass innerClass2 = outClass.new InnerClass();
        innerClass2.methodInner();
 
    }
}

【注意事项】


1. 外部类中的任何成员都可以在实例内部类方法中直接访问


2. 实例内部类所处的位置与外部类成员位置相同,因此也受public、private等访问限定符的约束


3. 在实例内部类方法中访问同名的成员时,优先访问自己的,如果要访问外部类同名的成员,必须:外部类名 称.this.同名成员 来访问


4. 实例内部类对象必须在先有外部类对象前提下才能创建(静态的可直接通过类名访问)


5. 实例内部类的非静态方法中包含了一个指向外部类对象的引用


6. 外部类中,不能直接访问实例内部类中的成员,如果要访问必须先要创建内部类的对象。

2.2 静态内部类

被static修饰的内部成员类称为静态内部类。

class OutClass {
    private int a;
    static int b;
    public void methodA(){
        a = 10;
        System.out.println(a);
    }
    public static void methodB(){
        System.out.println(b);
    }
    // 静态内部类:被static修饰的成员内部类
    static class InnerClass{
        public void methodInner(){
// 在内部类中只能访问外部类的静态成员
// a = 100; // 编译失败,因为a不是静态类成员变量
            b =200;
//methodA(); // 编译失败,因为methodA()不是类成员方法
//            methodB();
        }
    }
    public static void main(String[] args) {
// 静态内部类对象创建 & 成员访问
        OutClass.InnerClass innerClass = new OutClass.InnerClass();
        innerClass.methodInner();
    }
}

【注意事项】

1.在静态内部类中只能访问外部类中的静态成员(如果想访问,必须实例化外部类对象)

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

3.外部类可直接访问静态内部类所有成员变量与方法

2.3 局部内部类

定义在外部类的方法体或者{}中,该种内部类只能在其定义的位置使用,一般使用的非常少,此处简单了解下语法格式。

public class OutClass {
    int a = 10;
    public void method(){
    int b = 10;
// 局部内部类:定义在方法体内部
// 不能被public、static等访问限定符修饰
class InnerClass{
    public void methodInnerClass(){
        System.out.println(a);
        System.out.println(b);
    }
}
// 只能在该方法体内部使用,其他位置都不能用
    InnerClass innerClass = new InnerClass();
    innerClass.methodInnerClass();
    }
    public static void main(String[] args) {
//     OutClass.InnerClass innerClass = null; 编译失败
    }
}

【注意事项】

1. 局部内部类只能在所定义的方法体内部使用

2. 不能被public、static等修饰符修饰

3. 编译器也有自己独立的字节码文件,命名格式:外部类名字$数字内部类名字.class

4. 几乎不会使用

2.4 匿名内部类

interface First{
    void func();
}
 
class Second implements First{
    @Override
        public void func(){
            System.out.println("hello word");
        }
    }
public class Test {
    public static void main(String[] args) {
//此时是和上述的Second等价的,若想访问接口中的成员,可在大括号后通过.访问
        new First(){
            @Override
            public void func() {
                System.out.println("akai");
            }
        }.func();
    }
}


相关文章
|
监控 安全 Linux
龙蜥社区及阿里云CentOS迁移方案|飞天技术沙龙-CentOS 迁移替换专场
本次分享的主题是龙蜥社区及阿里云 CentOS 迁移方案|飞天技术沙龙- CentOS 迁移替换专场,由阿里云产品专家周絮分享。主要分为三个部分: 1.背景介绍 2.方案选型 3.迁移支持
327 0
|
分布式计算 算法 大数据
探索操作系统的核心:调度与内存管理机制
【10月更文挑战第11天】 本文深入探讨了操作系统中两大核心功能——调度与内存管理机制。通过分析调度算法、进程状态转换及内存分配策略等关键方面,揭示了它们如何共同维护系统性能和稳定性。旨在为读者提供对操作系统内部运作的深刻理解,同时引起对优化策略的思考。
450 5
|
算法 数据可视化 前端开发
第三代软件开发-加载本地字体库
欢迎来到我们的 QML & C++ 项目!这个项目结合了 QML(Qt Meta-Object Language)和 C++ 的强大功能,旨在开发出色的用户界面和高性能的后端逻辑。 在项目中,我们利用 QML 的声明式语法和可视化设计能力创建出现代化的用户界面。通过直观的编码和可重用的组件,我们能够迅速开发出丰富多样的界面效果和动画效果。同时,我们利用 QML 强大的集成能力,轻松将 C++ 的底层逻辑和数据模型集成到前端界面中。 在后端方面,我们使用 C++ 编写高性能的算法、数据处理和计算逻辑。C++ 是一种强大的编程语言,能够提供卓越的性能和可扩展性。我们的团队致力于优化代码,减少资
|
Kubernetes Nacos 容器
nacos注册不上
我正在使用开源的Nacos,并已在Kubernetes中部署了Nacos服务,通过端口映射可在集群外访问Nacos控制台。Kubernetes使用NodePort类型暴露了8848、9848、9849、7848和9555端口,但在尝试注册时遇到问题,出现“Client not connected, current status: STARTING”的错误,导致启动失败。
235 1
|
机器学习/深度学习 人工智能 自动驾驶
5G NR:下一代移动通信的基石
5G NR:下一代移动通信的基石
1804 1
|
算法 网络性能优化
Audio Over IP的PTP时钟初探
Audio Over IP的PTP时钟初探
290 0
|
Java 测试技术 API
详解单元测试问题之Mockito的注入过程如何解决
详解单元测试问题之Mockito的注入过程如何解决
486 1
|
监控 网络协议 Linux
在Linux中,如何查看当前系统每个 IP 的连接数?
在Linux中,如何查看当前系统每个 IP 的连接数?
|
消息中间件 Java Kafka
Docker 安装 kafka
Docker 安装 kafka
317 0