4

我知道并阅读了抽象类和接口,但我从未理解过的一点是,不能实例化的类有什么用。我可以使用普通类和虚拟方法而不是抽象类吗?当我实例化基类时会发生什么?

4

4 回答 4

21

当您有一组公共功能要在派生类之间共享时,您通常会使用抽象类。也就是说,您不能使用接口,因为您想提供一些默认功能。

看看System.IO.Stream课堂。此类提供一些通用的基本功能,但需要特定类型的流实现一些成员才能使其发挥作用。这些成员也被标记abstract为 ,这向编译器和运行时表明没有合适的基类实现。派生抽象类的非抽象类必须覆盖所有继承的抽象成员,就像实现接口的类必须实现接口上定义的所有成员一样。

在流示例中,Read()方法是抽象的,但ReadByte()方法不是——因为ReadByte()可以通过调用Read(). (虽然不是最佳的,这就是为什么ReadByte()是虚拟的,因此可以选择提供更有效的实现。)虚拟成员是不同的,因为它们确实有一个实现,但可以选择被覆盖。抽象成员默认没有实现,必须被覆盖。

换句话说,抽象类上的方法可以使用该类上的其他抽象成员,即使它们没有实现!这是因为需要派生的非抽象类提供实现——保证在调用方法时存在实现。这类似于您如何使用接口的成员,即使成员在接口上没有实现,因为它保证实现接口的对象必须实现其所有成员。

子类喜欢MemoryStreamFileStream覆盖所有抽象方法以形成一个具体的类,并且它们可以被实例化。但是,您可以将它们存储在Stream引用变量中,并将它们视为通用的“黑盒”流。这允许您声明一个接受Stream对象的方法,而您不必关心它实际上是哪种流。

Stream foo = new Stream();       // Invalid, Stream is abstract.
Stream foo = new MemoryStream(); // Valid.

所以,现在总结一下你在标题中提出的问题的答案。抽象类不能被实例化,因为它可能包含抽象的成员并且没有实现。抽象类的使用是双重的:首先,被子类化并允许子类共享某些成员的公共实现,其次,允许通过对抽象类的引用来使用子类的任何对象的实例。

于 2013-05-02T15:57:14.153 回答
0

抽象和接口使您可以共享一些通用逻辑,但您不能直接实例化它们中的任何一个

要添加到 cdhowie 答案,接口和抽象类之间最相关的区别是:

  • 从抽象类继承会迫使子类受到层次结构链的影响。使用接口实现它的不同类彼此完全松散。

  • 使用抽象类,您可以拥有具有逻辑的方法或属性,也就是说,一些代码在抽象类中自行实现。在接口中没有代码或逻辑,因此迫使实现者编写所有逻辑

于 2013-05-02T16:04:11.460 回答
0

抽象类和接口类是语言特性,它们都提供了一些编译时规则和一些运行时规则来帮助设计。到目前为止,实例化抽象类或接口类,它肯定不可能使用编译器,如果要使用 C++ 的汇编语言或说 C# 或 Java 的中间语言代码/字节码进行编程,那么也可以实例化它们,但我不确定这一点。因为在运行时,抽象类和接口类都有一个类型对象。

于 2013-05-02T16:14:35.913 回答
0

抽象类非常有用,而且都是关于设计的。例如,如果您有一个名为 Shape 的抽象基类,它具有诸如“绘制”和“移动”之类的功能。然后继承 Shape 类以创建“Circle”类和“Square”类。

继承的类都有 Draw 和 Move 函数。Move 可能在子类使用的基类中具有功能,但绘制功能由每个子类处理。

当你实例化一个圆形和方形时,只有一个“形状”对象是没有意义的。

希望有帮助。

于 2013-05-02T16:01:51.377 回答