Java学习笔记:内部类,静态内部类,匿名内部类

news2025/1/11 20:54:12

这是本人学习的总结,主要学习资料如下

  • 疯狂Java讲义第三版,李刚编,电子工业出版社出版

目录

  • 1、内部类
    • 1.1、内部类简介
    • 1.2、内部类与外部类的关系和区别:
    • 1.3、内部类的语法
  • 2、 非静态内部类
  • 3、静态内部类
  • 4、匿名内部类


1、内部类

1.1、内部类简介

  • 什么是内部类:一般类是一个独立的程序单元,但某些情况下会把一个类定义在另一个类的内部,这时在内部的类就是内部类。
  • 什么情况下使用内部类:如果一个内部类只在某个外部类中可以使用到,一旦这个内部类离开这个外部类后就没有意义,那么这个内部类就可以定义成内部类。比如创建一个Cow类,这个类需要CowLeg类。而CowLeg类离开Cow类没什么意义。Cow类又不希望CowLeg类被其他人访问,那么CowLeg就适合设为Cow类的内部类。

1.2、内部类与外部类的关系和区别:

  • 内部类可以看成是外部类的一个成员变量(但有不同):内部类也不能完全看成一个普通的成员,内部类中非静态内部类和静态内部类 拥有的访问权限并不相同,还需要具体区分。
  • 内部类可以使用private,protected和static修饰,而外部类不可以:先说外部类为什么不能使用。外部类应该是作为一个独立的程序单元运行,设为private(任何外部都不能调用)完全没意义,类加载器都不能找到它让它独立运行;protected(子类可调用)也没意义,只有子类能调用没什么用;static表示这个变量,方法或类属于某一个类(上一级),但是普通的类的上一级是一个包,普通的类加上static意义上说不过去,所以普通的类不能加static。
  • 非静态内部类不能拥有静态对象。

1.3、内部类的语法

只要在一个类的花括号内定义,那就是内部类。比如下面的代码。

public class Outer{
    public class inner{
        ...
    }
    ...
}

内部类甚至可以定义在类的方法中,那种叫局部内部类,作用于也只在那个方法中。

下面开始详细说非静态内部类的和静态内部类。



2、 非静态内部类

拥有内部类的类,在编译时产生的.class文件比较特殊,比如下面的代码

public class Outer {
    private class Inner{

    }
    public static void main(String[] args) {

    }
}

产生的.class文件如下图所示
Alt

除了外部类本身会产生的.class文件以外,还会产生 外部类名$内部类名.class文件。

前面说非静态内部类可以直接访问外部类的私有成员,这是因为非静态内部类的对象中保存了关于外部类对象的引用,如下图所示

Alt

CowLeg是非静态内部类,Cow是外部类。上面时是两个类的实例对象,可以看出,在CowLeg类对象中有一个Cow.this的引用帮助CowLeg类对象访问Cow类的成员。

当内部类想访问某个变量时,系统是先在内部类中查找有没有该变量名,如果没有再转到外部类中搜索,如果还是没找到,那时候才会报编译错误。

当内部类的变量和外部类的变量重名时,可以使用this,和外部类类名.this作为区分。

非静态内部类必须寄生于某一个外部类对象,也就是说,当存在一个非静态内部类实例时,一定有一个被他寄生的外部类实例,反之则不一定。所以,非静态内部类可以直接访问外部类的成员,因为外部类实例一定存在;而外部类不能直接访问非静态内部类成员,因为内部类对象可能不存在(可能还没实例化,所以不能直接访问。要先实例化内部类,才能通过内部类实例访问内部类成员)。

所以,静态代码块和静态方法中不能使用非静态的内部类,因为静态的代码块和方法属于类本身,而非静态内部类属于类实例,如果静态代码块和方法使用到非静态内部类,那计算机会糊涂,不知道要使用哪一个内部类实例,所以静态代码快和静态方法不能使用非静态内部类。

非静态内部类里不能使用静态的任何东西,包括方法,代码块,变量等。这个按道理来说应该是可以使用的,但这就是规定,不用去理解。

下面讲一下内部类的使用。这得分两种情况

  • 在外部类内部使用内部类:在外部类内部使用内部类和一般情况下使用普通的类是相同的,可以直接new 一个内部类。唯一需要注意的是不要再静态的部分使用非静态内部类。
  • 在外部类外部使用内部类:因为非静态内部类一定是依附于某个外部类的实例的,所以在外部使用非静态内部类时一定要通过外部类的实例去调用非静态内部类的构造器。这时候,非静态内部类前面的修饰,private,默认,protected,和public就是在这里起作用,这些修饰符表示的意义和他们修饰普通变量是一样的。private表示该内部类只能在外部类内部使用,默认就只能在外部类的包中使用,其它的也一样。

3、静态内部类

静态内部类就是在前面加上static。静态内部类也就是属于类本身,而不属于类的实例,那它不能调用外部类的非静态成员。其它的没什么好说的和非静态内部类一样。


4、匿名内部类

匿名内部类是需要重点关注的,因为理解了匿名内部类才能理解Java8的新特性lambda表达式

匿名内部类比较实用,它比较符合当初设计内部类时的场景。就是在某些场合需要使用类,但这个类只使用一次,以后再也不会使用了。

匿名内部类的语法

new 接口或父类构造器(实参列表){
    成员变量
}

匿名内部类因为是只使用一次,所以会在定义时就创建对象,所以匿名内部类不能是抽象的类。从上面的语法可以看出,匿名内部类因为没有类名,所以也不能定义构造器,只能通过初始化代码块来代替构造器。

匿名内部类最常用的场景是需要创建某个接口的对象。比如某个方法的参数正好需要一个接口的实例,专门定义一个类去实现这个接口没必要,这时候可以用匿名内部类。代码如下所示

interface Product{
    public double getPrice();
    public String getName();
}
public class Outer {
    public void anonymousTest(Product p){
        System.out.println("购买" + p.getName() + "花费了" + p.getPrice());
    }
    public static void main(String[] args) {
        Outer outer = new Outer();
        outer.anonymousTest(new Product()
        {
            @Override
            public double getPrice(){
                return 2.00;
            }
            @Override
            public String getName(){
                return "衣服";
            }
        });
    }
}

类Outer中方法anonymousTest的参数是接口Product,为了这个方法专门定义一个类去实现Product没什么必要,所以这里就用匿名内部类去实现这个接口来满足这个anonymousTest的需要。

匿名内部类继承父类时会涉及到参数列表的问题,这里的参数列表表示调用父类的构造器。

比如下面的代码

abstract class AnonymousTest{
    public AnonymousTest(){}
    public AnonymousTest(int a){
        System.out.println("进入了有a的构造器");
    }
}
public class Outer {
    public void test(AnonymousTest a){
        System.out.println("匿名类测试");
    }
    public static void main(String[] args) {
        Outer outer = new Outer();
        outer.test(new AnonymousTest()
        {
            //donothing
        });
        int a = 2;
        outer.test(new AnonymousTest(2) {
            //donothing
        });
    }
}

输出结果如下

Alt

使用匿名内部类时,还需要注意一点,就是匿名内部类使用到匿名类外部的变量时,那个变量会被自动加上final修饰符,比如下面的代码。

interface AnonymousTest{
    void test();
}
public class Outer {
    public void test(AnonymousTest a){
        System.out.println("匿名类测试");
    }
    public static void main(String[] args) {
        int age = 0;
        Outer outer = new Outer();
        outer.test(new AnonymousTest() {
            @Override
            public void test() {
                System.out.println(age);
            }
        });
        //这句代码会报错
        age = 4;
    }
}

Java8会为变量自动加上final,但是在java8之前匿名内部类引用类外变量之前必须为变量加上final,否则会报错。

为什么变量必须是final,其实是因为如果不加final,编译器实现起来会很困难,具体相关内容这里不赘述。

理解了匿名内部类以后,就能更好的理解lambda表达式,具体可看这篇文章。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/431536.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

“链引擎”入驻案例 | 每天超过35万条存证上链,长安链支撑链上价值流动

引言 长安链“链引擎”计划(Powered by Chainmaker)(简称:PBC计划)是由长安链生态联盟发起的一项应用赋能计划,旨在以长安链技术体系为核心支撑,汇聚产业各方力量,为应用方提供技术、品牌、生态等支持&…

Keil系列教程03_主窗口和工具栏详细说明

1写在前面 本文先让大家简单认识一下Keil的主窗口界面,然后再进一步认识Keil的文件、编译和调试工具栏。 Toolbars工具栏就是在菜单下面的两行快捷图标按钮,这些快捷按钮之所以在工具栏里面,在于它们使用的频率较高。比如保存按钮、编译按钮…

ChatGPT智能AI对话软件

ChatGPT智能AI的市场前景非常广阔,因为随着人工智能技术的不断发展和应用,人们对于智能AI对话系统的需求也越来越大。未来,智能AI对话系统将在各个领域得到广泛应用,例如智能客服、智能家居、自动驾驶等等,这些都有助于…

STM32 HAL库PID控制电机 第二章 TB6612FNG芯片驱动GB37-520电机

STM32 HAL库PID控制电机 第二章 TB6612FNG芯片驱动GB37-520电机(HAL库) 1 电路图 2 TB6612简介 TB6612是双驱动,可同时驱动两个电机 STBY:接单片机的IO口清零电机全部停止,置1通过AIN1 AIN2,BIN1,BIN2 来控制正反转…

linux下静态库和动态库的制作

一.静态库的制作 linux下库的命名规则:在linux下以libXXX.a为命名,lib(library)前缀是固定的,代表这个是库。接下来介绍静态库的制作流程。 1.1通过gcc编译获得.o文件 一般源程序经过预处理完成头文件和宏的展开&am…

运行时内存数据区之虚拟机栈——局部变量表

这篇内容十分重要,文字也很多,仔细阅读后,你必定有所收获! 基本内容 与程序计数器一样,Java虚拟机栈(Java Virtual Machine Stack)也是线程私有的,它的生命周期与线程相同。虚拟机栈描述的是Java方法执行的线程内存模型&#xf…

我想知道,就目前形势而言,学java好还是C++好?

前言 就现实点看看,可以对比现在Java和C的市场占有率,可以看到,到目前为止,Java在国内编程语言的市场仍然是占据着大头,在招聘当中Java的人数占有率仍然是遥遥领先于C,Java目前开阔的市场以及其巨大的岗位…

阿里,字节,拼多多,B站挨个面试一遍,你们猜哪个待遇最高?

我面试的是软件测试岗位,去年中旬的时候从原来的公司离职了,不是工作不好,而是公司发展速度太慢,自己干了几年,也没有太大的成长。以我目前的工作经验和实力,我认为准备一两个月,进大厂不是什么…

VS2022下载安装与基本使用(写C语言)

最近遇到一种问题,就是想要写一写C语言的代码,但是网页编辑器功能不全,GCC需要安装Liunx系统,VS又体量太大过于复杂,用keil又需要连接硬件,所以比较纠结。 工作中通常使用的是Keil,但是如果有时…

有记忆功能的动态通讯录

目录 前言1.进行文件操作的改造1.1contact.h的改造1.2contact.c的改造1.3test.c的改造 2.带文件操作的动态通讯录源码2.1contact.h2.2contact.c2.3test.c 总结 前言 前面我们一起学习的动态通讯录,一退出此程序联系人的信息就不见了;学习了文件操作操作后…

cocos creator 中使用web worker

1.应用场景:一些阻塞线程的方法可以放到worker里面去执行,不影响主线程,避免页面卡顿。 啊,有人就会说了,setTimeout不就可以了吗,还有什么async... JS本身就是单线程设计的,不管你是setTimeo…

EIGRP配置邻居关系详解

1.2 EIGRP 邻居关系 1.2.1 实验目的 通过 EIGRP 邻居建立的相关实验,学习到如何调整 EIGRP 的 HELLO 和 HOLD 时间,使用 被动接口阻止不必要的邻居关系,认证 EIGRP 邻居,静态邻居的配置以及哪些参数影响 EIGRP 邻居建立。 1.2.…

【数据库】Redis数据类型详解

目录一、5种基本数据类型1. String2. List3. Hash4. Set5. ZSet二、3种特殊类型1. Bitmap2. HyperLogLog3. Geospatial index一、5种基本数据类型 Redis 共有 5 种基本数据结构:String(字符串)、List(列表)、Set&#…

【CVPR轻量级网络】- 追求更高的FLOPS(FasterNet)

文章目录 题目:摘要1 介绍CNN中FLOPs的计算 2 相关工作3 PConv和FasterNet的设计3.1 偏卷积作为基本算子(PConv)3.2 PConv后接PWConv3.3 FasterNet作为通用骨干 4实验 题目: Run, Don’t Walk: Chasing Higher FLOPS for Faster Neural Netw…

Android---Jetpack之Paging

目录 Paging 组件的意思 Paging 支持的架构类型 Paging 的工作原理 PositionalDataSource PagekeyedDataSource ItemKeyedDataSource BoundaryCallback Paging 组件的意思 分页加载是在应用程序开发过程中十分常见的需求,Paging 就是 Google 为了方便 Andr…

JAVA局域网监听软件的设计与开发

网络监听软件是提供给网络安全管理人员进行安全管理的工具,可以用来监视网络的状态、数据流动情况以及网络上传输的信息,以获取有用信息。作为黑客来说,通过网络监听可以获取其所需信息(比如密码等);对黑客…

初中级Android工程师如何快速成长寻求突破

前言 写这篇文章的初衷是看到很多同学在一家公司工作了三五年,因为技术没有得到提升而随着年龄的增长导致不敢提出涨薪和跳槽找工作。希望这篇文章能够给这些还是初中级Android工程师的朋友一些启发。 快速成长 我们在向领导提出加薪申请或者是准备跳槽到更大的平…

概率机器学习笔记

1.单变量高斯混合分布 原书对结果的得出没有给出解释,我比较困惑,网上找到了一篇推导的帖子,看完就明白了。 式2.49的解释: 红框即为关键处,这是显而易见的期望,不过是条件方差的期望: 该证明的作者&…

共阳(共阴)LED数码管编码交互演示

LED数码管原理 LED数码管有两大类,一类是共阴极接法,另一类是共阳极接法,共阴极就是各段的显示字码共用一个电源的负极,是高电平点亮,共阳极就是各段的显示字码共用一个电源的正极,是低电平点亮。只要控制…

WPF教程(一)---创建一个WPF程序基础知识

1.前言: 这篇主要讲WPF的开发基础,介绍了如何使用Visual Studio 2019创建一个WPF应用程序。 首先说一下学习WPF的基础知识: 1) 要会一门.NET所支持的编程语言--例如C#。 2) 会一点“标准通用标记语言”:WPF窗体程序使用的XAML语…