关于抽象类无法实例化,但是可以使用“new“来实例的疑问

博客讨论了抽象类无法直接实例化,但可以通过‘new’操作创建子类对象的现象。通过一个具体的Java代码示例和debug过程,解释了在调试时看到的“MainApp$1”等实际上是匿名内部类的实例,这些内部类继承自抽象类。这表明,虽然抽象类自身不能实例化,但它的子类(包括匿名内部类)可以,并且‘new’操作创建的是子类的对象,而非抽象类本身。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

关于抽象类无法实例化,但是可以使用"new"来实例的疑问.

背景:昨天下午回答一个实习生问的问题,大致的经过是这样的:

他问的:“为什么抽象类无法实例化,但是可以new出来一个实例,官方这么说是什么意思?而且去了各类论坛,回答的都是比较含糊其辞,切不到重点.说什么是面向对象设计的规范balabala…”.

我心理想,这貌似是一个老生常谈的问题…

我回答说:“我给你一个demo,带你debug一下,流程走完了,剩下的你就自己体会下”

标题定义一个抽象类Animal:

package com.example.abstractl;

public abstract class Animal {

	public abstract void eat(Food food);

}

定义Animal的两个子类:

TomCat:

package com.example.abstractl;

public class TomCat extends Animal {

	@Override
	public void eat(Food food) {
		// TODO Auto-generated method stub
		System.out.println("TomCat Eating -> " + food.getName());
	}
}

JerryMicky:

package com.example.abstractl;

public class JerryMicky extends Animal {

	@Override
	public void eat(Food food) {
		// TODO Auto-generated method stub
		System.out.println("JerryMicky Eating -> " + food.getName());
	}

}

Food:

package com.example.abstractl;

public class Food {
	
	private String name;

	public Food(String name) {
		this.name = name;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}
	
	
}

MainApp类:

package com.example.abstractl;

public class MainApp {
	public static void main(String[] args) {
		Animal animal1 = new Animal() {

			@Override
			public void eat(Food food) {
				// TODO Auto-generated method stub
				System.out.println(food.getName());
			}
		};
		animal1.eat(new Food("老八秘制肠粉"));

		Animal animal2 = new Animal() {

			@Override
			public void eat(Food food) {
				// TODO Auto-generated method stub
				System.out.println(food.getName());
			}
		};
		animal2.eat(new Food("老八秘制辣椒酱"));

		Animal animal3 = new Animal() {

			@Override
			public void eat(Food food) {
				// TODO Auto-generated method stub
				System.out.println(food.getName());
			}
		};
		animal3.eat(new Food("老八秘制灌汤包"));

		Animal animal4 = new Animal() {

			@Override
			public void eat(Food food) {
				// TODO Auto-generated method stub
				System.out.println(food.getName());
			}
		};
		animal4.eat(new Food("老八秘制卤煮"));

		Animal tomcat = new TomCat();
		tomcat.eat(new Food("三文鱼"));
		Animal jerryMicky = new JerryMicky();
		jerryMicky.eat(new Food("奶酪"));
	}
}

打个断点,运行debug
在这里插入图片描述
现在只要关注Debug栈帧窗口(绿色框),观察**局部变量表(Variables)**里的数据:Name列与Value列

此时注意看到debug栈帧窗口的Value列,有个"MainApp$1" ~ "MainApp4"对象.

“右键"其中的某一行,比如:“animal3”—选择"Open Declared Type Hierarchy”.

看到如下图的Type Hierarchy窗口(红色框):
在这里插入图片描述

“$” 这就表示个内部类.而且还是个匿名内部类,而且还是作为局部变量的匿名内部类.

写道最后:

其实这个demo,也许写的不严谨.但是也基本说明:

为什么"抽象类只能被子类化。它不能被实例化。抽象类可以包含抽象方法,即已声明但未实现的方法。然后,子类提供抽象方法的实现。"的官方说明

好像从直观上说:

可以通过关键字来new一个抽象类来生成一个实例,但其实不是的,生成的仅仅是它的子类的对象.”

(好久不用md格式编辑,有点忘了,排版见谅).

OK,解释到此结束.

### 抽象类实例化使用 #### 抽象类不可直接实例化的原理 抽象类由于包含未实现的方法,在Java中不能被直接实例化。这是因为如果允许创建抽象类的对象,则其中的抽象方法将无从调用,因为它们并没有具体实现[^1]。 ```java abstract class Animal { public abstract void makeSound(); // 抽象方法没有方法体 public void eat() { // 非抽象方法 System.out.println("Eating..."); } } ``` 尝试如下方式实例化`Animal`将会导致编译错误: ```java Animal animal = new Animal(); // 错误:无法实例化抽象类 ``` #### 正确使用抽象类的方式 通过继承来间接利用抽象类的功能是一个常见模式。子类需提供抽象方法的具体实现,并可重写其他非抽象成员函数。只有当所有的抽象方法都被覆盖之后,该子类才能成为具体的、能够实例化的类[^2]。 ```java class Dog extends Animal { @Override public void makeSound() { System.out.println("Bark!"); } } public static void main(String[] args){ Dog dog = new Dog(); dog.makeSound(); // 输出 "Bark!" dog.eat(); // 输出 "Eating..." } ``` 在此例子中,虽然`Dog`扩展自`Animal`并实现了其唯一的抽象方法`makeSound()`,因此可以正常地创建`Dog`类型的对象而不会违反任何规则。 #### 子类初始化过程中的父类处理机制 值得注意的是,即使抽象类本身不允许作为独立实体存在,但在派生层次结构里它仍然扮演着重要角色。每当一个新的子类对象被创建时,JVM会自动执行一系列操作以确保整个继承链上的所有组件都能得到适当配置。这包括但不限于设置默认值给字段、运行构造器逻辑等动作[^3]。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值