首页 > 编程语言 >《JavaSE》---19.<补充:内部类&匿名内部类>

《JavaSE》---19.<补充:内部类&匿名内部类>

时间:2024-07-20 22:56:54浏览次数:27  
标签:部类 内部 19 成员 OutClass --- static JavaSE public

目录

前言

一、内部类

1.1内部类的概念 

1.2内部类的分类

①实例内部类 

【注意事项】

②静态内部类(相比实例内部类用的更多)

③局部内部类(几乎不用)

④匿名内部类(在下面讲)

二、匿名内部类

2.1匿名内部类简单介绍


前言

本篇博客主要讲解Java基础语法中的

一、内部类

内部类的概念、内部类的分类实例内部类、【注意事项】、静态内部类(相比实例内部类用的更多)、局部内部类(几乎不用)、匿名内部类。


      大家好,本人是普通一本的在校大学生一枚,目前在学习java。之前也学了一段时间,但是没有发布博客。时间过的真的很快。我会利用好这个暑假,来复习之前学过的内容,并整理好之前写过的博客进行发布。如果博客中有错误或者没有读懂的地方。热烈欢迎大家在评论区进行讨论!!!

      喜欢我文章的兄弟姐妹们可以点赞,收藏和评论我的文章。喜欢我的兄弟姐妹们以及也想复习一遍java知识的兄弟姐妹们可以关注我呦,我会持续更新滴,并且追求完整。
望支持!!!!!!一起加油呀!!!!

语言只是工具,不能决定你好不好找工作,决定你好不好找工作的是你的能力!!!!!

学历本科及以上就够用了!!!!!!!!!!!!!!!!!!!!!!!!!!!!


一、内部类

1.1内部类的概念 

在 Java 中,可以将一个类定义在另一个类或者一个方法的内部,前者称为内部类,后者称为外部类。内部类也是封装的一种体现。

当一个事物的内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么这个内部的完整结构最好使用内部类。

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

 注:

内部类和外部类共用同一个java源文件,

但是经过编译之后,内部类会形成单独的字节码文件

一个类一个字节码文件

1.2内部类的分类

①实例内部类 

成员位置定义

也叫普通的成员内部类、非静态内部类。即未被static修饰的成员内部类。

public class OutClass {
    private int a = 1;
    public static int b = 2;
    public int c = 3;
    public void methodA(){
        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;
            //实例内部类中也可以有自己的普通成员变量、普通成员方法、构造方法
            public int d = 4;
            private int e = 5;
            //public static int f = 6;
            //但是不能有静态成员变量和成员方法。如果非要定义则可以用final修饰
            public static final int f = 6;

            methodA();
            methodB();
 
            // 如果外部类和实例内部类中具有相同名称成员时,优先访问的是内部类自己的
            c = 300;
            System.out.println(c);
 
            // 如果要访问外部类同名成员时候,必须:外部类名称.this.同名成员名字
            OutClass.this.c = 400;
            System.out.println(OutClass.this.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.methodA();
        outClass.methodB();
 
        System.out.println("=============实例内部类的访问=============");
        // 要访问实例内部类中成员,必须要创建实例内部类的对象
        // 而普通内部类定义与外部类成员定义位置相同,因此创建实例内部类对象时必须借助外部类
 
        // 创建实例内部类对象
        OutClass.InnerClass innerClass1 = new OutClass().new InnerClass();
 
        // 上述语法比较怪异,也可以先将外部类对象先创建出来,然后再创建实例内部类对象
        OutClass.InnerClass innerClass2 = outClass.new InnerClass();
        innerClass2.methodInner();
   }
}

1.在实例内部类中,我们可以有构造方法、普通成员变量、普通成员方法 

如何实例化内部类对象呢,我们需要通过外部类的引用如

OutClass.InnerClass innerClass1 = new OutClass().new InnerClass();

或者

OutClass outClass = new OutClass();
OutClass.InnerClass innerClass2 = outClass.new InnerClass();

2.不能有静态成员变量和成员方法。因为实例内部类是依赖于外部类对象的。而static修饰的变量是不依赖于对象的。因此这两者有矛盾。

如果非要有static修饰的成员变量。则可以用final修饰。因为被final修饰的是常量。常量不需要类加载。编译的时候就知道这个值是存在的。

 3. 在实例内部类方法中访问同名的成员时,优先访问自己的,

如果要访问外部类同名的成员 必须:

外部类名称.this.同名成员

【注意事项】

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

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

3. 实例内部类对象必须在先有外部类对象前提下才能创建

4. 实例内部类的非静态方法中包含了一个指向外部类对象的引用,这个引用就是this。(上面第三三点说的this)

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

②静态内部类(相比实例内部类用的更多)

成员位置定义

被static修饰,并且无需通过外部类就可以创建静态内部类对象。

public 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();   // 编译失败,因为methodB()不是类成员方法
            methodB();
       }
   }
 
    public static void main(String[] args) {
        // 静态内部类对象创建 & 成员访问
        OutClass.InnerClass innerClass = new OutClass.InnerClass();
        innerClass.methodInner();
   }
}

【注意事项】

1. 在静态内部类中只能访问外部类中的静态成员。

如果确实想访问外部类的普通成员,我们该如何做?

我们可以在静态内部类中实例化外部类对象。通过这个对象来访问外部类中的普通成员

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

③局部内部类(几乎不用)

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

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. 几乎不会使用  

④匿名内部类(在下面讲)

public class OutClass {
    // 成员位置定义:未被static修饰 --->实例内部类
    public class InnerClass1{
   }
 
    // 成员位置定义:被static修饰 ---> 静态内部类
    static class InnerClass2{
 
   }
 
 
    public void method(){
        // 方法中也可以定义内部类 ---> 局部内部类:几乎不用
        class InnerClass5{
 
       }
   }
}

根据内部类定义的位置不同,一般可以分为以下几种形式:

1. 成员内部类(普通内部类:未被static修饰的成员内部类 和 静态内部类:被static修饰的成员内部类)

2. 局部内部类(不谈修饰符)、匿名内部类

注意:

内部类其实日常开发中使用并不是非常多,大家在看一些库中的代码时候可能会遇到的比较多,日常开发中使用最多的是匿名内部类。 

二、匿名内部类

2.1匿名内部类简单介绍

匿名内部类涉及到了接口的知识。

我们可以认为,匿名内部类,实现了某个接口,并且重写了接口中的方法。

代码示例:

interface A{
    void testA();
}

public class Main {
    public static void main(String[] args) {
        new A(){    //我们可以理解为有一个类,实现了A接口,并且重写了A接口中的方法。
            @Override
            public void testA() {
                
            }
        };//匿名内部类,这个类没有名字
    }
}

我们可以理解为有一个类,实现了A接口,并且重写了A接口中的方法。 

1.并且在创建这个匿名内部类之后。我们可以在匿名内部类的结尾的 } 来调用匿名内部类中的方法。 

2.也可以通过这个接口接收这个类,再来调用匿名内部类中的方法。

interface A{
    void testA();
}

public class Main {
    public static void main(String[] args) {
        new A(){
            @Override
            public void testA() {
                System.out.println("你好,匿名内部类!");
            }
        }.testA();
    }
}
        A a = new A(){
            @Override
            public void testA() {
                System.out.println("你好,匿名内部类!");
            }
        };
        a.testA();

运行结果:

你好,匿名内部类!

我们在学习多线程的时候会经常用到这种匿名内部类的写法。         

注意事项:

在匿名内部类中,能够访问的成员变量是没有被修改的数据(默认在这里能访问的是被final修饰的。但也能访问没有被修改的成员变量)。有个专业术语叫变量的捕获

标签:部类,内部,19,成员,OutClass,---,static,JavaSE,public
From: https://blog.csdn.net/m0_73456341/article/details/140576790

相关文章

  • Nginx系列-3 servername优先级和location优先级和常用正则表达式
    1.正则表达式和分组由于Nginx配置文件中经常出现正则表达式,因此本章节专门对常见的正则表达式进行简单介绍。[1]开始与结束^表示匹配输入字符串的开始$表示匹配输入字符串的结束[2]匹配次数?表示匹配0次或者1次+表示匹配1次或多次*表示匹配0从或多次{n}匹配n......
  • 史上最强的管理监控工具之Ollama-webui
    目录欢迎关注微信公众号:数据科学与艺术Ollama-webui是一个管理和监控工具,用于管理和监控Ollama应用程序。以下是使用Ollama-webui的详细操作步骤:首先,确保你已经正确安装了Ollama和Ollama-webui。你可以从官方网站下载最新的版本并按照安装说明进行安装。安装完成......
  • Linux C++ 065-设计模式之组合模式
    LinuxC++065-设计模式之组合模式本节关键字:Linux、C++、设计模式、组合模式相关库函数:概念组合模式(CompositePattern),又叫做部分-整体模式,使得用户对单个对象和组合对象的使用具有一致性。它使我们树型结构的问题中,模糊了简单元素和复杂元素的概念,客户程序可以像处理......
  • Linux C++ 066-设计模式之访问者模式
    LinuxC++066-设计模式之访问者模式本节关键字:Linux、C++、设计模式、访问者模式相关库函数:概念在访问者模式(VisitorPattern)中,我们使用了一个访问者类,它改变了元素类的执行算法。通过这种方式,元素的执行算法可以随着访问者改变而改变。这种类型的设计模式属于行为型模......
  • Qt项目:基于Qt实现的网络聊天室---好友申请
    文章目录完善ClickedLabel好友申请美化界面本文介绍如何实现好友申请界面,其效果如下图所示在此之前需要先定义一个ClickedOnceLabel类,支持点击一次的label功能。接着新增一个ClickedOnceLabel类classClickedOnceLabel:publicQLabel{Q_OBJECTpublic:......
  • 玄机-第二章日志分析-mysql应急响应
    文章目录前言简介应急开始准备工作日志分析步骤1步骤2步骤3步骤4总结补充erro.log前言这里应急需要知道mysql提权的一些姿势,还有能够提权成功的前提。5金币就当复习一下了。这里考察的是mysql应急响应,我们应该是根据找flag的需求去就行,但是我做了之后发现......
  • React函数式组件---Hooks
            这里先简单复习一下函数式组件:没有自己的this,没有实例对象,不能使用三大属性中的state和refs,仅能用props;不能使用生命周期!1.ReactHook/Hooks是什么?(1).Hook是React16.8.0版本增加的新特性/新语法(2).可以让你在函数组件中使用state以及其他的React......
  • C++面向对象编程的一个核心概念--RAII
    RAII是"ResourceAcquisitionIsInitialization"(资源获取即初始化)的缩写,它是C++编程中的一种编程技术,用于管理资源的生命周期。RAII是C++面向对象编程的一个核心概念,它利用对象的构造函数和析构函数来自动管理资源,如内存、文件句柄、线程、互斥锁等。RAII的主要原则包括:1.*......
  • YOLOv8改进 | Neck | 注意力尺度序列融合的检测框架ASF-YOLO
    秋招面试专栏推荐:深度学习算法工程师面试问题总结【百面算法工程师】——点击即可跳转......
  • ABC 363 E - Sinking Land 题解
    用一个优先队列维护和海相邻的位置,每次海面上升就判断一下队列中海拔最低的那个位置会不会被淹没,如果会,就删除,同时它上下左右的位置也是和海相邻的(或者就在海里),把它们加进优先队列里,记得判断一下加入的格子曾经有没有被加入过队列,不要加重复了。点击开DconstintN=1099;int......