1核心库类之TPersistent类TPersistent = class(TObject)
 2核心库类之TPersistent类  private
 3核心库类之TPersistent类    procedure AssignError(Source: TPersistent);
 4核心库类之TPersistent类  protected
 5核心库类之TPersistent类    procedure AssignTo(Dest: TPersistent); virtual;
 6核心库类之TPersistent类    procedure DefineProperties(Filer: TFiler); virtual;
 7核心库类之TPersistent类    function  GetOwner: TPersistent; dynamic;
 8核心库类之TPersistent类  public
 9核心库类之TPersistent类    destructor Destroy; override;
10核心库类之TPersistent类    procedure Assign(Source: TPersistent); virtual;
11核心库类之TPersistent类    function  GetNamePath: string; dynamic;
12核心库类之TPersistent类  end;
     在TPersistent的声明中,有两个Public的方法(Destroy在此不讨论),其中GetNamePath是Delphi的集成开发环境内部使用的,VCL不推荐直接对它的调用。而Assign方法则是为完成对象复制而存在的,并且被声明为虚方法,以允许每个派生类定义自己的复制对象的方法。
    如果正在设计的类需要有这种允许对象复制的能力,则让类从TPersistent派生并重写Assign方法。
    如果没有重写Assign方法,则TPersistent的Assign方法会将复制动作交给源对象来 进行:
procedure TPersistent.Assign(Source: TPersistent);
begin
   if Source <> nil then
       Source.AssignTo(Self) // 调用源对象的AssignTo方法
   else
      AssignError(nil);
end;
可以在TPersistent类的声明的protected节中找到AssignTo方法的声明,它也是一个虚方法。
如果将复制动作交给源对象来完成,那么必须保证源对象的类已经重写了AssignTo方法,否则将抛出一个“Assign Error”异常:
 1核心库类之TPersistent类{ TPersistent }
 2核心库类之TPersistent类
 3核心库类之TPersistent类destructor TPersistent.Destroy;
 4核心库类之TPersistent类begin
 5核心库类之TPersistent类  RemoveFixups(Self);
 6核心库类之TPersistent类  inherited Destroy;
 7核心库类之TPersistent类end;
 8核心库类之TPersistent类
 9核心库类之TPersistent类procedure TPersistent.Assign(Source: TPersistent);
10核心库类之TPersistent类begin
11核心库类之TPersistent类  if Source <> nil then Source.AssignTo(Self) else AssignError(nil);
12核心库类之TPersistent类end;
13核心库类之TPersistent类
14核心库类之TPersistent类procedure TPersistent.AssignError(Source: TPersistent);
15核心库类之TPersistent类var
16核心库类之TPersistent类  SourceName: string;
17核心库类之TPersistent类begin
18核心库类之TPersistent类  if Source <> nil then
19核心库类之TPersistent类    SourceName := Source.ClassName else
20核心库类之TPersistent类    SourceName := 'nil';
21核心库类之TPersistent类  raise EConvertError.CreateResFmt(@SAssignError, [SourceName, ClassName]);
22核心库类之TPersistent类end;
23核心库类之TPersistent类
24核心库类之TPersistent类procedure TPersistent.AssignTo(Dest: TPersistent);
25核心库类之TPersistent类begin
26核心库类之TPersistent类  Dest.AssignError(Self);
27核心库类之TPersistent类end;
28核心库类之TPersistent类
29核心库类之TPersistent类procedure TPersistent.DefineProperties(Filer: TFiler);
30核心库类之TPersistent类begin
31核心库类之TPersistent类end;
32核心库类之TPersistent类
33核心库类之TPersistent类function TPersistent.GetNamePath: string;
34核心库类之TPersistent类var
35核心库类之TPersistent类  S: string;
36核心库类之TPersistent类begin
37核心库类之TPersistent类  Result := ClassName;
38核心库类之TPersistent类  if (GetOwner <> nil) then
39核心库类之TPersistent类  begin
40核心库类之TPersistent类    S := GetOwner.GetNamePath;
41核心库类之TPersistent类    if S <> '' then
42核心库类之TPersistent类      Result := S + '.' + Result;
43核心库类之TPersistent类  end;
44核心库类之TPersistent类end;
45核心库类之TPersistent类
46核心库类之TPersistent类function TPersistent.GetOwner: TPersistent;
47核心库类之TPersistent类begin
48核心库类之TPersistent类  Result := nil;
49核心库类之TPersistent类end;
    AssignError是一个private方法,仅仅用于抛出赋值错误的异常。
在TPersistent的声明中,GetOwner方法是被前面所述由Delphi内部使用的GetNamePath所调用。
最后还剩下一个虚方法DefineProperties(),它则是为TPersistent的另一个使命而存在:对象持久。一个对象要持久存在,就必须将它流化(Streaming),保存到一个磁盘文件(.dfm文件)中。TPersistent也使得其派生类具有这种能力,但它作为抽象类只是定义接口而并没有给出实现。可以看到,DefineProperties是一个空的虚方法:
procedure TPersistent.DefineProperties(Filer: TFiler);
begin
end;
这留待其派生类来实现。
对于对象持久的实现类,最典型的就是TComponent,每个组件都具有保存自己的能力。
2.持久化的核心类:TFiler、TReader、TWriter,并使用继承的方式来实现。
TFiler抽象类,定义持久化基本的服务接口,同是TReader、TWriter的抽象父类。

相关文章:

  • 2021-04-18
  • 2021-11-30
  • 2022-12-23
  • 2021-05-03
  • 2022-01-21
  • 2022-12-23
  • 2021-04-08
  • 2021-09-11
猜你喜欢
  • 2022-12-23
  • 2022-12-23
  • 2022-12-23
  • 2022-12-23
  • 2021-07-25
  • 2021-10-27
  • 2022-12-23
相关资源
相似解决方案