Java基础——非静态内部类的实例必须依赖外部类的实例存在

2024-04-23 09:28

本文主要是介绍Java基础——非静态内部类的实例必须依赖外部类的实例存在,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

背景

之前预习多态的时候,看到chatgpt3.5给的示例是经典的关于猫狗的举例。代码并不难,一看感觉就会,但是放到idea就开始报错。。。。搞了好久换了好几次方式,才突然反应过来,其实有些基础知识忘记了。。。。记录下。

过程

chatgpt3.5给的示例如下:

// Animal 类,作为基类
class Animal {// 发出声音的方法public void makeSound() {System.out.println("动物发出声音");}
}// Dog 类,继承自 Animal 类
class Dog extends Animal {// 重写 makeSound 方法public void makeSound() {System.out.println("汪汪汪!");}
}// Cat 类,继承自 Animal 类
class Cat extends Animal {// 重写 makeSound 方法public void makeSound() {System.out.println("喵喵喵!");}
}// 多态示例
public class Main {// 接受 Animal 对象并调用 makeSound 方法public static void animalSound(Animal animal) {animal.makeSound();}public static void main(String[] args) {// 创建 Dog 和 Cat 对象Dog dog = new Dog();Cat cat = new Cat();// 调用 animalSound 方法并传入不同的对象animalSound(dog); // 输出:"汪汪汪!"animalSound(cat); // 输出:"喵喵喵!"}
}

将上述代码放入idea中显示如下:

package com.zj.polymorphism;
/*** Java中多态使用示例* 多态定义:允许不同对象对同一消息做出不同的反应* 方法重现,方法重载,以及接口的实现体现了多态
* */
public class PolymorphismTest3 {// Animal 类,作为基类class Animal {// 发出声音的方法public void makeSound() {System.out.println("动物发出声音");}}// Dog 类,继承自 Animal 类class Dog extends Animal {// 重写 makeSound 方法public void makeSound() {System.out.println("汪汪汪!");}}// Cat 类,继承自 Animal 类class Cat extends Animal {// 重写 makeSound 方法public void makeSound() {System.out.println("喵喵喵!");}}// 多态示例// 接受 Animal 对象并调用 makeSound 方法public static void animalSound(Animal animal) {animal.makeSound();}public static void main(String[] args) {// 创建 Dog 和 Cat 对象Dog dog = new Dog();Cat cat = new Cat();// 调用 animalSound 方法并传入不同的对象animalSound(dog); // 输出:"汪汪汪!"animalSound(cat); // 输出:"喵喵喵!"}}

其实是会报错的,报错如下:
在这里插入图片描述

Inner classes cannot have static declarations:内部类不能有静态声明;

在这个地方idea提示其实有错误,更精确的说法是非静态内部类不能有静态声明。如果该内部类是静态的话,那么其内部是可以有静态声明的,是符合Java语法的。
那么就将这个Main内部类去掉就是了,单独放在外部类中,如下:
package com.zj.polymorphism;
/*** Java中多态使用示例* 多态定义:允许不同对象对同一消息做出不同的反应* 方法重现,方法重载,以及接口的实现体现了多态
* */
public class PolymorphismTest3 {// Animal 类,作为基类class Animal {// 发出声音的方法public void makeSound() {System.out.println("动物发出声音");}}// Dog 类,继承自 Animal 类class Dog extends Animal {// 重写 makeSound 方法public void makeSound() {System.out.println("汪汪汪!");}}// Cat 类,继承自 Animal 类class Cat extends Animal {// 重写 makeSound 方法public void makeSound() {System.out.println("喵喵喵!");}}// 多态示例// 接受 Animal 对象并调用 makeSound 方法public static void animalSound(Animal animal) {animal.makeSound();}public static void main(String[] args) {// 创建 Dog 和 Cat 对象Dog dog = new Dog();Cat cat = new Cat();// 调用 animalSound 方法并传入不同的对象animalSound(dog); // 输出:"汪汪汪!"animalSound(cat); // 输出:"喵喵喵!"}}

可是如此写之后还是会报错,报错如下:
在这里插入图片描述


cannot be referenced from a static context:不能够从静态上下文中引用。
就是从这里开始让我感到很疑惑的,因为一眼看上去,好像没有问题,就是在main方法中实例化Dog和Cat两个对象,然后将两个对象放到静态的animalSound方法中,传递的实例对象是子类Dog和Cat两个对象,而不是要求的Animal对象,这样的方式就已经体现了多态了(允许不同的对象对同一个消息做出不同的响应)。但是这个报错说明这样的写法有问题。
从报错上来说,不能够从静态上下文中引用,完全没头绪的。为什么不能引用,平常调用的时候不都是可以在静态Main主方法中实例化对象的吗????

总之懵逼了很久,才突然想到难道是因为静态的内部方法无法实例化同样的内部类吗?然后将main方法放到单独一个类中跟平常一样调用,结果还是失败的。。。。。这说明跟是否为静态的内部方法无关。。。
后来问chatgpt3.5联网版本才反应过来,这里的Dog和Cat是非静态内部类!!!
非静态内部类是必须要依赖外部类实例才能存在的!!!无论该非静态内部类是处于外部类的内部还是外部都是要先实例化外部类才能拿到该非静态内部类的实例
也就说,要实例化Dog和Cat对象,即使在外部类的内部,要实例化内部类也需要先实例化外部类才能得到非静态内部类的对象。
如下:
 public static void main(String[] args) {//方式一Animal dog = new PolymorphismTest2().new Dog();Animal cat = new PolymorphismTest2().new Cat();animalSound(dog);animalSound(cat);//方式二/*PolymorphismTest2 polymorphismTest = new PolymorphismTest2();polymorphismTest.animalSound(dog);polymorphismTest.animalSound(cat);*/}

按照上面方式写就不会报错。。。。

内省

真的,折腾这么久敢情就是一个非静态的内部类的实例化调用需要先调用外部类。。。。这个知识点并不是说我不知道,而是忽视了。。。因为如果平常是在外部的一个类调用另一个类的非静态内部类的时候这种使用场景就会自然而然的先实例化外部类再去实例化内部类,这个过程都没有感知的,因为有习惯了。但是在这里,因为是在同一个外部类的Main主方法内,想当然的以为不需要再实例化外部类就可以调用内部类了。。。。所以可以得知有些语法规则是要彻底的执行到底的,跟所处的范围是没有关系的,需要严格的执行。

这篇关于Java基础——非静态内部类的实例必须依赖外部类的实例存在的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/928410

相关文章

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

Java中List的contains()方法的使用小结

《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

Spring Boot 配置文件之类型、加载顺序与最佳实践记录

《SpringBoot配置文件之类型、加载顺序与最佳实践记录》SpringBoot的配置文件是灵活且强大的工具,通过合理的配置管理,可以让应用开发和部署更加高效,无论是简单的属性配置,还是复杂... 目录Spring Boot 配置文件详解一、Spring Boot 配置文件类型1.1 applicatio

C# WinForms存储过程操作数据库的实例讲解

《C#WinForms存储过程操作数据库的实例讲解》:本文主要介绍C#WinForms存储过程操作数据库的实例,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、存储过程基础二、C# 调用流程1. 数据库连接配置2. 执行存储过程(增删改)3. 查询数据三、事务处

Java中StopWatch的使用示例详解

《Java中StopWatch的使用示例详解》stopWatch是org.springframework.util包下的一个工具类,使用它可直观的输出代码执行耗时,以及执行时间百分比,这篇文章主要介绍... 目录stopWatch 是org.springframework.util 包下的一个工具类,使用它

Java进行文件格式校验的方案详解

《Java进行文件格式校验的方案详解》这篇文章主要为大家详细介绍了Java中进行文件格式校验的相关方案,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、背景异常现象原因排查用户的无心之过二、解决方案Magandroidic Number判断主流检测库对比Tika的使用区分zip