我非常厌倦了看到:
function descendent.doSomething() : TList;
begin
inherited;
end;
当我为基类得到这个时:
function descendent.doSomething() : TList;
begin
result := nil;
end;
并且不愿意结束:
function descendent.doSomething() : TList;
begin
end;
然后想知道为什么有些东西不起作用。
我喜欢使用抽象函数,因为编译器会告诉您是否因为没有实现某些函数而导致抽象错误。
我的问题是,因为我仍然是一个相对较新的Delphi程序员而且我从未需要维护任何8年的事情,是否值得以这种方式花费时间prune your code(即删除函数只是在它们中继承并将基类函数从抽象更改为具体)
答案 0 :(得分:7)
一如既往地取决于问题。我使用接口来定义类集的用户界面。至少当我知道我将有多个底层实际类的实现时。例如,你可以这样:
IAllInterfaced = interface(IInterface)
procedure ImplementMeEverywhere_1(const Params: TParams);
procedure ImplementMeEverywhere_2(const Params: TParams);
procedure ImplementMeEverywhere_3(const Params: TParams);
end;
TAllInterfaced_ClassA = class(TInterfacedObject, IAllInterfaced)
public
procedure ImplementMeEverywhere_1(const Params: TParams);
procedure ImplementMeEverywhere_2(const Params: TParams);
procedure ImplementMeEverywhere_3(const Params: TParams);
end;
TAllInterfaced_ClassB = class(TInterfacedObject, IAllInterfaced)
public
procedure ImplementMeEverywhere_1(const Params: TParams);
procedure ImplementMeEverywhere_2(const Params: TParams);
procedure ImplementMeEverywhere_3(const Params: TParams);
end;
在这里你没有共同的祖先。每个类只实现接口,并且没有公共基类形式的公共底层结构。如果实现是如此不同以至于它们不共享任何东西,那么这是可能的,但是他自己接口。您仍然需要使用相同的接口,以便与派生类的用户保持一致。
第二个选项是:
IAllAbstract = interface(IInterface)
procedure ImplementMeEverywhere_1(const Params: TParams);
procedure ImplementMeEverywhere_2(const Params: TParams);
procedure ImplementMeEverywhere_3(const Params: TParams);
end;
TAllAbstract_Custom = (TInterfacedObject, IAllAbstract)
private
...
public
procedure ImplementMeEverywhere_1(const Params: TParams); virtual; abstract;
procedure ImplementMeEverywhere_2(const Params: TParams); virtual; abstract;
procedure ImplementMeEverywhere_3(const Params: TParams); virtual; abstract;
end;
TAllAbstract_ClassA = class(TAllAbstract_Custom)
public
procedure ImplementMeEverywhere_1(const Params: TParams); override;
procedure ImplementMeEverywhere_2(const Params: TParams); override;
procedure ImplementMeEverywhere_3(const Params: TParams); override;
end;
TAllAbstract_ClassB = class(TAllAbstract_Custom)
public
procedure ImplementMeEverywhere_1(const Params: TParams); override;
procedure ImplementMeEverywhere_2(const Params: TParams); override;
procedure ImplementMeEverywhere_3(const Params: TParams); override;
end;
这里有一个所有类的基类。在该类中,您可以拥有公共属性或事件其他类等...但是所有过程都标记为抽象,因为它们不执行任何常见任务。 Abstract确保它们将在派生类中实现,但是您不需要在每个类中实现“FieldA”,只能在“TAllAbstract_Custom”中实现它。这确保了使用DRY原则。
最后一个选项是:
IAllVirtual = interface(IInterface)
procedure ImplementMeEverywhere_1(const Params: TParams);
procedure ImplementMeEverywhere_2(const Params: TParams);
procedure ImplementMeEverywhere_3(const Params: TParams);
end;
TAllVirtual_Custom = (TInterfacedObject, IAllVirtual)
private
...
public
procedure ImplementMeEverywhere_1(const Params: TParams); virtual;
procedure ImplementMeEverywhere_2(const Params: TParams); virtual;
procedure ImplementMeEverywhere_3(const Params: TParams); virtual;
end;
TAllVirtual_ClassA = class(TAllVirtual_Custom)
public
procedure ImplementMeEverywhere_1(const Params: TParams); override;
procedure ImplementMeEverywhere_2(const Params: TParams); override;
procedure ImplementMeEverywhere_3(const Params: TParams); override;
end;
TAllVirtual_ClassB = class(TAllVirtual_Custom)
public
procedure ImplementMeEverywhere_1(const Params: TParams); override;
procedure ImplementMeEverywhere_2(const Params: TParams); override;
procedure ImplementMeEverywhere_3(const Params: TParams); override;
end;
这里所有派生类都有一个共同的基本虚拟过程。这可确保您不必在派生类的级别上实现每个过程。您只能覆盖代码的某些部分或根本不覆盖。
当然这都是边缘情况,beetwen有空间。你可以混合使用这些概念。
请记住:
很抱歉答案很长,但我不能在这里给出一个简单的解释,因为没有。这一切都取决于手头的问题。它是派生类共有多少以及它们的实现有多么不同之间的平衡。
答案 1 :(得分:1)
如果代码非常简单并且您发现它很难阅读并且容易出错,那么它可能很难阅读并且容易出错。 (另一方面,如果代码复杂并且您发现它很难阅读,那可能是您缺乏经验。但不是这样的。)您可能很好地重构它现在,虽然这个问题在你脑海中仍然很新鲜。
答案 2 :(得分:1)
是的,修剪你的代码。
它使您的其他代码更容易阅读(正如您已经提到的,更容易看到实际覆盖了哪些方法)。作为一个额外的好处,更容易在父类中更改方法的签名:想象一下,您决定将另一个参数传递给虚方法;您对父类进行了更改,然后您需要为从给定父类继承的每个子类重复相同的更改。那时你不想要伪造被称为“继承”的伪造方法!