我们可以实例化一个抽象类吗?

在我的一次面试中,有人问我:“我们是否可以实例化一个抽象类?”

我的回答是:“没有。我们不能”。但是,面试官告诉我:“错了,我们可以。”

我对此进行了一些争论。然后他让我自己在家试试。

abstract class my {
public void mymethod() {
System.out.print("Abstract");
}
}


class poly {
public static void main(String a[]) {
my m = new my() {};
m.mymethod();
}
}

在这里,我正在创建我的类的实例和调用抽象类的方法。有人能给我解释一下吗?我的面试真的错了吗?

310892 次浏览

这里,我正在创建我的类的实例

不,您不是在这里创建抽象类的实例。相反,您是在创建抽象类的匿名子类实例。然后调用指向子类对象抽象类引用上的方法。

这种行为清楚地列在章节# 15.9.1中:-

如果类实例创建表达式在类体中结束,则 被实例化的类是一个匿名类。然后:< / p >

  • 如果T表示一个类,则声明一个由T命名的类的匿名直接子类。类型为编译时错误 由T表示的class是最终类
  • 如果T表示一个接口,则声明Object的一个匿名直接子类,该子类实现了以T命名的接口。
  • 在任何一种情况下,子类的主体都是类实例创建表达式中给出的ClassBody。
  • 被实例化的类是匿名子类。

我特别强调。

此外,在JLS -第12.5节中,您可以阅读到对象创建过程。我在这里引用其中的一段话:-

每当创建一个新的类实例时,就分配内存空间 为该类中声明的所有实例变量留出空间 类的每个超类中声明的所有实例变量 类类型,包括所有可能被隐藏的实例变量

在对新创建对象的引用返回之前 结果,则处理指定的构造函数以初始化new

你可以在我提供的链接上阅读完整的程序。


要实际看到被实例化的类是匿名子类,您只需要编译两个类。假设你把这些类放在两个不同的文件中:

My.java:

abstract class My {
public void myMethod() {
System.out.print("Abstract");
}
}

Poly.java:

class Poly extends My {
public static void main(String a[]) {
My m = new My() {};
m.myMethod();
}
}

现在,编译两个源文件:

javac My.java Poly.java

现在在你编译源代码的目录中,你会看到以下类文件:

My.class
Poly$1.class  // Class file corresponding to anonymous subclass
Poly.class

看到那个类Poly$1.class。它是编译器创建的类文件,对应于你使用下面的代码实例化的匿名子类:

new My() {};

很明显,有一个不同的类被实例化了。只是,这个类只有在编译器编译后才会被赋予名称。

一般来说,类中的所有匿名子类都将以这种方式命名:

Poly$1.class, Poly$2.class, Poly$3.class, ... so on

这些数字表示这些匿名类在外围类中出现的顺序。

上面实例化了一个匿名内部类,它是my抽象类的子类。严格来说,它并不等同于实例化抽象类本身。OTOH,每个子类实例都是其所有超类和接口的实例,因此大多数抽象类确实是通过实例化它们的一个具体子类来实例化的。

如果面试官只是说“错了!”而没有解释,并给出了这个例子作为一个独特的反例,我认为他不知道自己在说什么。

= my() {};意味着有一个匿名实现,而不是对象的简单实例化,它应该是:= my()。你永远不能实例化一个抽象类。

抽象类不能被实例化,但可以被子类化。# EYZ0

最好的例子是

虽然日历类有一个抽象方法getInstance(),但当你说Calendar calc=Calendar.getInstance();

# EYZ0“;

事实上,匿名内部类型 允许您创建抽象类的无名称子类和这个实例。

你可以观察到:

  1. 为什么poly扩展了my?这没用……
  2. 编译的结果是什么?三个文件:my.classpoly.classpoly$1.class
  3. 如果我们可以这样实例化一个抽象类,我们也可以实例化一个接口……奇怪的……


我们可以实例化一个抽象类吗?

不,我们不能。我们可以做的是,创建一个匿名类(这是第三个文件)并实例化它。


那么超类实例化呢?

抽象超类不是由我们实例化的,而是由java实例化的。

编辑:请他测试一下

public static final void main(final String[] args) {
final my m1 = new my() {
};
final my m2 = new my() {
};
System.out.println(m1 == m2);


System.out.println(m1.getClass().toString());
System.out.println(m2.getClass().toString());


}

输出是:

false
class my$1
class my$2

你可以简单地回答,只用一行

没有,你永远不能实例化抽象类

但是,面试官仍然不同意,那么你可以告诉他/她

你所能做的就是创建一个匿名类。

根据匿名类,类在同一位置/行声明和实例化

所以,面试官可能会对你的自信程度和你对OOPs的了解程度感兴趣。

abstract class可以在每个人回答时实例化,这是一个公认的事实。

当程序定义匿名类时,编译器实际上创建了一个具有不同名称的新类(具有EnclosedClassName$n模式,其中n是匿名类号)

所以如果你反编译这个Java类,你会发现如下代码:

my.class

abstract class my {
public void mymethod()
{
System.out.print("Abstract");
}
}

Poly $1.class(匿名类的生成类)

class poly$1 extends my
{
}

ploly.cass

public class poly extends my
{
public static void main(String[] a)
{
my m = new poly.1(); // instance of poly.1 class NOT the abstract my class


m.mymethod();
}
}

技术答案

抽象类不能被实例化——这是通过定义和设计实现的。

摘自JLS,第8章。类:

命名类可以声明为抽象类(§8.1.1.1),并且必须声明 抽象,如果没有完全实现;这样的类是不可能的

.

来自JSE 6 java文档的Classes.newInstance():

InstantiationException -如果这个类表示一个抽象类,一个接口,一个数组 Class,一个基本类型,或void;或者类没有null构造函数;或者如果 实例化失败是由于其他原因

当然,您可以实例化抽象类的具体子类(包括匿名子类),也可以对抽象类型的对象引用进行类型转换。

从不同的角度看这个问题——团队合作社会智力:

在现实世界中,当我们处理复杂的技术和法律规范时,这种技术误解经常发生。

在这里,“人际交往能力”可能比“技术能力”更重要。如果你试图证明自己的观点,那么理论上你可能是正确的,但你也可能在战斗中造成更大的伤害/损害“面子”/制造敌人。以和解和理解的方式解决你们的分歧。谁知道呢——也许你“都是对的”,但对术语的含义略有不同??

谁知道呢——虽然不太可能,但也有可能面试官故意引入一个小冲突/误解,把你置于一个具有挑战性的环境中,看看你在情感和社交方面的表现如何。对同事要有礼貌和建设性,听从前辈的建议,在面试结束后通过电子邮件或电话解决任何挑战/误解。这表明你很积极,注重细节。

扩展类并不意味着实例化类。实际上,在您的示例中,您正在创建子类的一个实例。

我非常确定抽象类不允许初始化。所以,我会说不:你不能实例化一个抽象类。但是,您可以扩展/继承它。

你不能直接实例化一个抽象类。但这并不意味着您不能间接获得类的实例(实际上不是原始抽象类的实例)。我的意思是你不能实例化原始的抽象类,但是你可以:

  1. 创建一个空类
  2. 从抽象类继承
  3. 实例化派生类

因此,您可以通过派生类实例访问抽象类中的所有方法和属性。

关于抽象类

  • 不能创建抽象类的对象
  • 可以创建变量(可以像数据类型一样运行)
  • 如果子对象不能重写父对象的至少一个抽象方法,那么子对象也将成为抽象对象
  • 如果没有子类,抽象类是无用的

抽象类的目的是表现得像基类。在继承层次结构中,你会看到抽象类位于顶部。

不,你不能实例化一个抽象类。我们只实例化匿名类。在抽象类中,我们只声明抽象方法和定义具体方法。

不可能实例化一个抽象类。 你真正能做的是,在一个抽象类中实现一些通用方法,而让其他方法未实现(将它们声明为抽象),并让具体的派生程序根据需要实现它们。 然后您可以创建一个工厂,该工厂返回这个抽象类的实例(实际上是他的实现者)。然后在工厂中决定选择哪个实现者。这被称为工厂设计模式:

   public abstract class AbstractGridManager {
private LifecicleAlgorithmIntrface lifecicleAlgorithm;
// ... more private fields


//Method implemented in concrete Manager implementors
abstract public Grid initGrid();


//Methods common to all implementors
public Grid calculateNextLifecicle(Grid grid){
return this.getLifecicleAlgorithm().calculateNextLifecicle(grid);
}


public LifecicleAlgorithmIntrface getLifecicleAlgorithm() {
return lifecicleAlgorithm;
}
public void setLifecicleAlgorithm(LifecicleAlgorithmIntrface lifecicleAlgorithm) {
this.lifecicleAlgorithm = lifecicleAlgorithm;
}
// ... more common logic and getters-setters pairs
}

具体实现者只需要实现声明为抽象的方法,但可以访问在抽象类中那些类中实现的逻辑,这些类不是声明为抽象的:

public class FileInputGridManager extends AbstractGridManager {


private String filePath;


//Method implemented in concrete Manager implementors
abstract public Grid initGrid();


public class FileInputGridManager extends AbstractGridManager {


private String filePath;


//Method implemented in concrete Manager implementors
abstract public Grid initGrid();


public Grid initGrid(String filePath) {
List<Cell> cells = new ArrayList<>();
char[] chars;
File file = new File(filePath); // for example foo.txt
// ... more logic
return grid;
}
}

最后,工厂看起来是这样的:

public class GridManagerFactory {
public static AbstractGridManager getGridManager(LifecicleAlgorithmIntrface lifecicleAlgorithm, String... args){
AbstractGridManager manager = null;


// input from the command line
if(args.length == 2){
CommandLineGridManager clManager = new CommandLineGridManager();
clManager.setWidth(Integer.parseInt(args[0]));
clManager.setHeight(Integer.parseInt(args[1]));
// possibly more configuration logic
...
manager = clManager;
}
// input from the file
else if(args.length == 1){
FileInputGridManager fiManager = new FileInputGridManager();
fiManager.setFilePath(args[0]);
// possibly more method calls from abstract class
...
manager = fiManager ;
}
//... more possible concrete implementors
else{
manager = new CommandLineGridManager();
}
manager.setLifecicleAlgorithm(lifecicleAlgorithm);
return manager;
}
}

AbstractGridManager的接收者将调用他身上的方法并获得在具体的下降器中实现的逻辑(部分在抽象类方法中实现),而不知道他得到的具体实现是什么。这也称为控制反转或依赖注入。

技术部分在其他答案中已经很好地涵盖了,主要以:
结束 “他错了,他什么都不懂,让他加入SO,把一切都弄清楚:)”< / p >

我想说明一个事实(在其他回答中已经提到过),这可能是一个stress-question,是许多面试官更多了解你以及你如何应对困难和不寻常情况的重要工具。通过给你错误的代码,他可能想看看你是否会反驳。要知道在类似的情况下,你是否有信心站起来对抗你的前辈。

附注:我不知道为什么,但我有一种感觉,面试官已经读了这篇文章。

不,我们不能创建抽象类的对象,而是创建抽象类的引用变量。引用变量用于引用派生类(抽象类的子类)的对象。

下面的例子说明了这个概念

abstract class Figure {


double dim1;


double dim2;


Figure(double a, double b) {


dim1 = a;


dim2 = b;


}


// area is now an abstract method


abstract double area();


}




class Rectangle extends Figure {
Rectangle(double a, double b) {
super(a, b);
}
// override area for rectangle
double area() {
System.out.println("Inside Area for Rectangle.");
return dim1 * dim2;
}
}


class Triangle extends Figure {
Triangle(double a, double b) {
super(a, b);
}
// override area for right triangle
double area() {
System.out.println("Inside Area for Triangle.");
return dim1 * dim2 / 2;
}
}


class AbstractAreas {
public static void main(String args[]) {
// Figure f = new Figure(10, 10); // illegal now
Rectangle r = new Rectangle(9, 5);
Triangle t = new Triangle(10, 8);
Figure figref; // this is OK, no object is created
figref = r;
System.out.println("Area is " + figref.area());
figref = t;
System.out.println("Area is " + figref.area());
}
}

在这里,我们可以看到不能创建类型为Figure的对象,但可以创建类型为Figure的引用变量。这里我们创建了一个类型为Figure的引用变量,而Figure类引用变量用于引用Rectangle和Triangle类的对象。

你可以说:
我们不能实例化一个抽象类,但是我们可以使用new关键字来创建一个匿名类实例,只需在抽象类的末尾添加{}作为实现体。

实际上,我们不能直接创建抽象类的对象。我们创建的是一个抽象调用的引用变量。引用变量用于引用继承抽象类的类的对象,即抽象类的子类。