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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

每天认识几个maven依赖(ActiveMQ+activemq-jaxb+activesoap+activespace+adarwin)

八、ActiveMQ 1、是什么? ActiveMQ 是一个开源的消息中间件(Message Broker),由 Apache 软件基金会开发和维护。它实现了 Java 消息服务(Java Message Service, JMS)规范,并支持多种消息传递协议,包括 AMQP、MQTT 和 OpenWire 等。 2、有什么用? 可靠性:ActiveMQ 提供了消息持久性和事务支持,确保消

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟 开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚 第一站:海量资源,应有尽有 走进“智听