GNU Pascal follows the object model of Borland Pascal 7.0. The BP object extensions are almost fully implemented into GPC. This includes inheritance, virtual and non-virtual methods, constructors, destructors, pointer compatibility, extended `New' syntax (with constructor call and/or as a Boolean function), extended `Dispose' syntax (with destructor call).
The Borland object model is different from the ISO draft, but it will not be too difficult now to implement that too (plus the Borland Delphi Object Extensions which are quite similar to the ISO draft).
The syntax for an object type declaration is as follows:
program ObjectDemo; type Str100 = String (100); FooParentPtr = ^FooParent; FooPtr = ^Foo; FooParent = object constructor Init; destructor Done; virtual; procedure Bar (c: Real); virtual; function Baz (b, a, z: Char) = s: Str100; { not virtual } end; Foo = object (FooParent) x, y: Integer; constructor Init (a, b: Integer); destructor Done; virtual; procedure Bar (c: Real); virtual; { overrides `FooParent.Bar' } z: Real; { GPC extension: data fields after methods } function Baz: Boolean; { new function } end; constructor FooParent.Init; begin WriteLn ('FooParent.Init') end; destructor FooParent.Done; begin WriteLn ('I''m also done.') end; procedure FooParent.Bar (c: Real); begin WriteLn ('FooParent.Bar (', c, ')') end; function FooParent.Baz (b, a, z: Char) = s: Str100; begin WriteStr (s, 'FooParent.Baz (', b, ', ', a, ', ', z, ')') end; constructor Foo.Init (a, b: Integer); begin inherited Init; x := a; y := b; z := 3.4; FooParent.Bar (1.7) end; destructor Foo.Done; begin WriteLn ('I''m done.'); inherited Done end; procedure Foo.Bar (c: Real); begin WriteLn ('Foo.Bar (', c, ')') end; function Foo.Baz: Boolean; begin Baz := True end; var Ptr: FooParentPtr; begin Ptr := New (FooPtr, Init (2, 3)); Ptr^.Bar (3); Dispose (Ptr, Done); New (Ptr, Init); with Ptr^ do WriteLn (Baz ('b', 'a', 'z')) end.
Remarks:
A pointer to FooParent may be assigned the address of a Foo object. A FooParent formal var parameter may get a Foo object as the actual parameter. In such cases, a call to a virtual method calls the child's method, whereas a call to a non-virtual method selects the parent's one:
var MyFooParent: FooParentPtr; SomeFoo: Foo; [...] SomeFoo.Init (4, 2); MyFooParent := @SomeFoo; MyFooParent^.bar (3.14); { calls `foo.bar' } MyFooParent^.baz ('b', 'a', 'z'); { calls `fooParent.baz' } if SomeFoo.baz then { calls `foo.baz' } WriteLn ('Baz!');
In a method, an overwritten method of a parent object can be called either prefixing it with the parent type name, or using the keyword inherited:
procedure Foo.Bar (c: Real); begin z := c; inherited bar (z) { or: FooParent.Bar (z) } end;
Use FooParent.bar (z) if you want to be sure that this method is called, even if somebody decides not to derive foo directly from fooParent but to have some intermediate object. If you want to call the method bar of the immediate parent – whether it be fooParent or whatever – use inherited bar (z).
To allocate an object on the heap, use New in one of the following manners:
var MyFoo: FooPtr; [...] New (MyFoo, Init (4, 2)); MyFooParent := New (FooPtr, Init (4, 2))
The second possibility has the advantage that MyFoo needn't be a FooPtr but can also be a FooParentPtr, i.e. a pointer to an ancestor of foo.
Destructors can and should be called within Dispose:
Dispose (MyFooParent, Fini)